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“网 站 开发 案例 课堂 ”系列 图 书 是 专门 为 网 页 设计 和 动态 网 站 开发 初学 者 量 身 定制 的 一 套 
学 习 用 书 。 整 套 书 涵盖 网 页 设计 、 网 站 开发 、 数 据 库 设计 等 方面 。 整 套 书 具 有 以 下 几 个 特点 。 


前 沿 科 技 

无 论 是 网 站 建设 、 数 据 库 设计 还 是 HTML 5、CSS 3、JavaScript、PHP， 我 们 都 精 选 较为 
前 沿 或 者 用 户 群 最 大 的 领域 推进 ， 帮 助 大 家 认识 和 了 解 最 新 动态 。 

权威 的 作者 团队 


组 织 国家 重点 实验 室 和 资深 应 用 专家 联手 编著 该 套图 书 ， 融 合 丰富 的 教学 经 验 与 优秀 的 


学 习 型 案例 设计 

以 技术 的 实际 应 用 过 程 为 主线 ， 全 程 采 用 图 解 和 同步 多 媒体 结合 的 教学 方式 ， 生 动 、 直 
观 、 全 面 地 剖析 使 用 过 程 中 的 各 种 应 用 技能 ， 降 低 难 度 ， 提 升学 习 效 率 。 
为 什么 要 写 这 样 一 本 书 


PHP 是 世界 上 最 为 流行 的 Web 开发 语言 之 一 。 目 前 学 习 和 关注 PHP 的 人 越 来 越 多 ， 而 
很 多 PHP 的 初学 者 都 苦于 找 不 到 一 本 通俗 易 懂 、 容 易 入 门 和 案例 实用 的 参考 书 。 为 此 ， 作 者 
组 织 有 丰富 经 验 的 开发 人 员 编 写 了 这 本 书 。 通 过 本 书 的 实 训 ， 读 者 可 以 很 快 地 学 会 开发 动态 
网 站 ， 提 高 职业 化 能 力 ， 从 而 帮助 解决 公司 与 求职 者 的 双重 需求 问题 。 


本 书 特色 
= 零 基础 、 入 门 级 的 讲解 


无 论 您 是 否 从 事 计算 机 相关 行业 ， 无 论 您 是 否 接触 过 PHP+MySQL 动态 网 站 开发 ， 都 能 
从 本 书 中 找到 最 佳 起 点 。 


于 超 多 、 实 用 、 专 业 的 范例 和 项 目 

本 书 在 编排 上 紧密 结合 深入 学 习 网 页 制作 技术 的 先后 过 程 ， 从 PHP 7 的 基本 概念 开始 ， 
逐步 带领 大 家 深入 地 学 习 各 种 应 用 技巧 ， 侧 重 实战 技能 ， 使 用 简单 易 懂 的 实际 案例 进行 分 析 
和 操作 指导 ， 让 读者 读 起 来 简明 轻松 ， 操 作 起 来 有 章 可 循 。 

”随时 检测 自己 的 学 习 成 果 

内 容 讲解 章节 最 后 的 “疑难 解 惑 ” 板 块 ， 均 根据 本 章 内 容 精 选 而 成 ， 从 而 帮助 读者 解决 
自学 过 程 中 最 常见 的 疑难 问题 。 





PHP+MySQL 动态 网 站 开发 
案例 课堂 (第 2 版 ) B… 


于 ”细致 入 微 、 贴 心 提示 


本 书 在 讲解 过 程 中 ， 在 各 章 中 使 用 了 “注意 ”“ 提 示 ”“ 技 巧 ” 等 小 贴 士 ， 使 读者 在 学 
习 过 程 中 更 清楚 地 了 解 相 关 操 作 、 理 解 相关 概念 ， 并 轻松 掌握 各 种 操作 技巧 。 


”专业 创作 团队 和 技术 支持 


您 在 学 习 过 程 中 遇 到 任何 问题 ， 均 可 加 入 QQ 群 (案例 课堂 VIP)451102631 进行 提问 ， 专 
家 人 员 会 在 线 答疑 。 


超 值 资源 大 放送 
全程 同 步 教学 录像 


涵盖 本 书 所 有 知识 点 ， 详 细 讲解 每 个 实例 及 项 目的 过 程 及 技术 关键 点 。 可 以 使 读者 比 看 书 更 
轻松 地 掌握 书 中 所 有 的 动态 网 站 开发 知识 ， 而 且 扩展 的 讲解 部 分 使 您 得 到 比 书 中 更 多 的 收获 。 


超 多 容量 王牌 资源 


赠送 大 量 王牌 资源 ， 包 括 本 书 实例 源 代码 、 教 学 幻灯 片 、 本 书 精品 教 
学 视频 、16 个 经 典 项 目 开发 完整 源 代 码 、 常 用 SQL 语句 速 查 手册 、 
MySQLi 函数 速 查 手册 、PHP 7 废弃 特性 速 查 手册 、PHP 7 的 新 功能 速 查 
手册 、PHP 常用 函数 速 查 手册 、PHP 网 站 开发 工程 师 面试 技巧 、PHP 网 
站 开发 工程 师 常见 面试 题 、 优 秀 网 站 开发 工程 师 之 路 一 一 网 站 开发 经 验 及 
技巧 大 汇总 等 。 读 者 可 以 通过 QQ 群 (案例 课堂 VIP)451102631 获取 赠送 
资源 ， 也 可 以 扫描 二 维 码 ， 下 载 本 书 资源 。 


读者 对 象 


没有 任何 网 页 设计 基础 的 初学 者 。 

有 一 定 的 PHP 7 基础 ， 想 精通 PHP 7 动态 网 站 开发 的 人 员 。 
有 一 定 的 PHP 7 网 页 设计 基础 ， 没 有 项 目 经 验 的 人 员 。 
正在 进行 毕业 设计 的 学 生 。 

大 专 院 校 及 培训 学 校 的 老师 和 学 生 。 


创作 团队 


本 书 由 刘 春 茂 编著 ， 参 加 编写 的 人 员 还 有 刘 玉 萍 、 张 金 伟 、 薄 娟 、 周 佳 、 付 红 、 李 园 、 
郭 广 新 、 侯 永 岗 、 王 攀登 、 刘 海松 、 孙 若 淞 、 王 月 娇 、 包 慧 利 、 陈 伟 光 、 胡 同 夫 、 王 伟 、 展 
娜 娜 、 李 琪 、 梁 云 梁 和 周 浩 浩 。 在 编写 过 程 中 ， 我 们 竭尽 所 能 地 将 最 好 的 讲解 呈现 给 读者 ， 
但 也 难免 有 玻 漏 和 不 妥 之 处 ， 敬 请 不 音 指正 。 若 您 在 学 习 中 遇 到 困难 或 疑问 ， 或 有 任何 建 
议 ， 可 写 信人 至 信箱 357975357@qq.com。 
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第 1 章 
初 识 庐山 真面目 


一 一 走 进 PHP 7 
的 世界 


在 学 习 PHP 之 前 ， 读 者 需要 了 解 PHP 的 基本 概念 、PHP 的 特点 、PHP 开发 常 
用 工具 、PHP 7 的 新 特征 和 如 何 学 习 PHP 7 等 知识 。 本 章 主 要 讲述 PHP 的 入 门 知 
识 。 通 过 本 章 的 学 习 ， 读 者 能 够 对 PHP 先 有 一 个 初步 的 了 解 。 
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1.1 认识 PHP 
PHP 语言 与 其 他 语言 有 什么 不 同 ? 读者 首先 需要 理解 PHP 的 概念 和 发 展 历程 。 
1.1.1 什么 是 PHP 


PHP 全 名 为 Personal Home Page， 是 英文 Hypertext Preprocessor( 超 级 文本 预 处 理 语 言 ) 的 
别名 。PHP 作为 在 服务 器 端 执行 的 嵌入 HIML 文档 的 脚本 语言 ， 其 风格 类 似 于 C 语言 ， 被 运 
用 于 动态 网 站 开发 。PHP 借鉴 了 C 语言 和 Java 语言 等 的 部 分 语法 ， 并 有 自己 的 特性 ， 使 Web 
开发 者 能 够 快速 地 编写 动态 生成 页 面 的 脚本 。 

对 初学 者 而 言 ，PHP 的 优势 是 可 以 快速 入 门 。 与 其 他 编程 语言 相 比 ，PHP 是 将 程序 能 入 
到 HTML 文档 中 去 执行 的 ， 执 行 效率 比 完全 生成 HTML 标记 的 方式 要 高 许多 。PHP 还 可 以 执 
行 编译 后 的 代码 。 编 译 可 以 起 到 加 密 和 优化 代码 运行 的 作用 ， 使 代码 运行 得 更 快 。 另 外 ， 
PHP 具有 非常 强大 的 功能 ， 能 够 实现 所 有 的 CGI 功能 ， 而 且 支 持 几乎 所 有 流行 的 数据 库 和 操 
作 系 统 。 最 重要 的 是 ，PHP 还 可 以 用 C、C++ 进 行程 序 扩展 。 


1.1.2 PHP 的 发 展 历程 


在 当今 诸多 Web 开发 语言 中 ，PHP 是 比较 出 众 的 一 种 。 与 其 他 脚本 语言 不 同 ，PHP 是 经 
过 全 世界 免费 代码 开发 者 共同 努力 ， 才 发 展 到 今天 的 规模 的 。 要 想 了 解 PHP， 首 先 应 该 从 它 
的 发 展 历程 谈 起 。 

1994 年 ，Rasmus Lerdorf 首次 开发 了 PHP 程序 设计 语言 。1995 年 6 月 ，Rasmus Lerdorf 
在 Usenet 新 闻 组 comp.infosystems.www.authoring.cgi 上 发 布 了 PHP 1.0 声明 。 这 个 早期 版 本 
提供 了 访客 留言 本 、 访 客 计 数 器 等 简单 的 功能 。 

1995 年 ， 第 2 版 的 PHP 问 市 ， 定 名 为 PHP/FI(Form Interpreter)。 在 这 一 版 本 中 ， 加 入 了 
可 以 处 理 更 复杂 的 嵌入 式 标签 语言 的 解析 程序 ， 同 时 加 入 了 对 数据 库 MySQL 的 支持 。 自 
此 ， 葛 定 了 PHP 在 动态 网 页 开发 上 的 影响 力 。 自 从 PHP 加 入 了 这 些 强大 的 功能 以 后 ， 它 的 使 
用 量 猛 增 。 据 初步 统计 ，1996 年 年 底 ， 有 15000 个 Web 网 站 使 用 了 PHP/FI; 而 1997 年 中 
期 ， 这 一 数字 超过 了 50000。 

PHP 前 两 个 版 本 的 成 功 ， 让 其 设计 者 和 使 用 者 对 PHP 的 未 来 充满 了 信心 。1997 年 ， 
Zeev Suraski 及 Andi GutmansPHP 加 入 了 开发 小 组 ， 他 们 自愿 重新 编写 了 底层 的 解析 引擎 ， 又 
有 其 他 很 多 人 也 自愿 加 入 了 PHP 的 工作 ， 使 得 PHP 成 为 真正 意义 上 的 开源 项 目 。 

1998 年 6 月 ， 发 布 了 PHP 3.0 声明 。 在 这 一 版 本 中 ，PHP 可 以 跟 Apache 服务 器 紧密 地 结 
合 ; 再 加 上 它 不 断 地 更 新 及 加 入 新 的 功能 ， 且 支持 几乎 所 有 主流 和 非 主流 数据 库 ， 拥 有 非常 
高 的 执行 效率 。 这 些 优势 使 1999 年 使 用 PHP 的 网 站 数量 超过 了 150000。 

PHP 经 过 3 个 版 本 的 演化 ， 已 经 变 成 一 种 非常 强大 的 Web 开发 语言 。 这 种 语言 非常 容易 
使 用 ， 而 且 它 拥 有 一 个 强大 的 类 库 ， 类 库 的 命名 规则 也 十 分 规范 。 新 手 就 算 对 一 些 函数 的 功 
能 不 了 解 ， 也 可 以 通过 函数 名 猜测 出 来 。 这 使 得 PHP 十 分 容易 学 习 ， 而 且 PHP 程序 可 以 直接 


使 用 HTML 编辑 器 来 处 理 。 因 此 ，PHP 变 得 非常 流行 。 有 很 多 大 的 门户 网 站 都 使 用 了 PHP 作 
为 自己 的 Web 开发 语言 ， 如 新 浪 网 等 。 

2000 年 5 月 ， 推 出 了 划时代 的 版 本 PHP 4。 它 使 用 了 一 种 “编译 -执行 ”模式 。 核 心 引擎 
更 加 优越 ， 提 供 了 更 高 的 性 能 ， 而 且 还 包含 了 其 他 一 些 关键 功能 ， 比 如 支持 更 多 的 Web 服务 
器 、HTTP Sessions 支持 、 输 出 缓存 、 更 安全 的 处 理 用 户 输入 的 方法 和 一 些 新 的 语言 结构 。 

2004 年 7 月 ，PHP 5.0 发布 。 该 版 本 以 Zend 引擎 [为 引擎 ， 并 且 加 入 了 新 功能 ， 如 PHP 
Data Objects(PDO)。PHP 5.0 版 本 强化 更 多 的 功能 。 首 先 ， 完 全 实现 面向 对 象 ， 提 供 名 为 PHP 
兼容 模式 的 功能 。 其 次 是 XML 功能 ，PHP 5.0 版 本 支持 可 直观 地 访问 XML 数据 、 名 为 
SimpleXML 的 XML 处 理 用 户 界 面 。 同 时 还 强化 了 XMLWeb 服务 支持 ， 而 且 标 准 支 持 SOAP 
扩展 模块 。 

PHP 到 目前 的 最 新 版 本 是 PHP 7， 功 能 更 加 强大 ， 执 行 效率 更 高 。 本 书 将 针对 PHP 7 版 
本 ， 讲 解 PHP 的 实用 技能 。 


1.1.3 PHP 语言 的 优势 


PHP 能 够 迅速 发 展 ， 并 得 到 广大 使 用 者 的 喜爱 ， 其 主要 原因 是 PHP 不 仅 有 一 般 脚 本 都 具 
备 的 功能 ， 而 且 有 它 自身 的 优势 ， 具 体 特 点 如 下 。 

(1) 源 代码 完全 开放 。 所 有 的 PHP 源 代码 事实 上 都 可 以 得 到 。 读 者 可 以 通过 Internet 获得 
所 需要 的 源 代码 ， 快 速 修 改 和 利用 。 

(2) 完全 免费 。 与 其 他 技术 相 比 ，PHP 本 身 是 免费 的 。 使 用 PHP 进行 Web 开发 无 须 支付 
任何 费用 。 

(3) 语法 结构 简单 。PHP 结合 了 C 语言 和 Perl 语言 的 特色 ， 编 写 简单 ， 方 便 易 懂 。 可 以 
嵌入 HTML 语言 中 ， 相 对 于 其 他 语言 编辑 简单 ， 实 用 性 强 ， 更 适合 初学 者 学 习 。 

(4) 跨 平台 性 强 。PHP 是 服务 器 端 脚本 ， 可 以 运行 于 UNIX、Linux、Windows 环境 下 。 

(5) 效率 高 。PHP 消耗 非常 少 的 系统 资源 ， 并 且 程 序 开发 快 ， 运 行 速度 快 。 

(6) 强大 的 数据 库 支 持 。PHP 支持 目前 所 有 的 主流 和 非 主流 数据 库 ， 使 PHP 的 应 用 对 象 
非常 广泛 。 

(7) 面向 对 象 。 在 PHP 中 ， 面 向 对 象 方面 有 了 很 大 改进 ， 现 在 PHP 完全 可 以 用 来 开发 大 
型 商业 程序 了 。 





1.2 PHP 能 干什么 


初学 者 也 许 会 有 疑问 ，PHP 到 底 能 做 什么 呢 ? 下 面 就 来 介绍 PHP 的 应 用 领域 。 

1. 作为 服务 器 端 脚 本 

PHP 最 主要 的 应 用 领域 是 作为 服务 器 端 脚本 。 服 务 器 端 脚本 的 运行 需要 具备 3 项 配置 : 
PHP 解析 器 、Web 浏览 器 和 Web 服务 器 。 在 Web 服务 器 上 安装 并 配置 PHP， 然 后 用 Web 浏 
览 器 访问 PHP 程序 ， 获 得 输出 。 在 学 习 的 过 程 中 ， 读 者 只 要 在 本 机 上 配置 Web 服务 器 ， 即 可 
浏览 制作 的 PHP 页 面 。 
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2. 作为 命令 行 脚本 


命令 行 脚本 与 服务 器 端 脚本 不 同 ， 编 写 的 命令 行 脚本 并 不 需要 任何 服务 器 或 浏览 器 。 在 
命令 行 脚本 模式 下 ， 只 需要 PHP 解析 器 执行 即 可 。 这 些 脚 本 被 用 在 Windows 和 Linux 平台 下 
作为 日 常 运行 脚本 ， 也 可 以 用 来 处 理 简 单 的 文本 。 


3. 用 来 编写 桌面 应 用 程序 


PHP 在 桌面 应 用 程序 的 开发 中 并 不 常用 ， 但 是 如 果 用 户 希 望 在 客户 端 应 用 程序 中 使 用 
PHP 编写 图 形 界面 应 用 程序 ， 可 以 通过 PHP-GTK 来 编写 这 些 程序 。PHP-GTK 是 PHP 的 扩 
展 ， 并 不 包含 在 标准 的 开发 包 中 ， 开 发 用 户 需要 单独 编译 它 。 


1.3 ”常用 的 开发 利器 


可 以 编写 PHP 代码 的 工具 很 多 ， 常 见 的 有 Dreamweaver、PHPEdit、PHPed 和 FrontPage 
等 ， 甚 至 用 Word 和 记事 本 等 常用 工具 也 可 以 书写 PHP 源 代 码 。 


1.3.1 PHP 代码 开发 工具 


常见 的 PHP 代码 开发 工具 如 下 。 
1. PHPEdit 


PHPEdit 是 一 款 Windows 下 优秀 的 PHP 脚本 IDE( 集 成 开发 环境 )。 该 软件 为 快速 、 便 捷 
地 开发 PHP 脚本 提供 了 多 种 工具 ， 功 能 包括 : 语法 关键 词 高 亮 ， 代码 提示 、 浏 览 ， 集成 PHP 
调试 工具 ; 帮助 生成 器 ; 自 定 义 快捷 方式 ，150 多 个 脚本 命令 ;键盘 模板 ; 报告 生成 器 ; 快速 
标记 和 插件 等 。 


2. gPHPedit 


gPHPedit 是 在 Linux 下 十 分 流行 的 免费 的 PHP 编辑 器 ， 它 小 巧 而 功能 强大 。 它 是 以 
Linux 下 的 gedit 文本 编辑 器 为 基础 ， 专 门 设计 ， 用 于 编辑 PHP 和 HTML 的 编辑 器 ， 可 以 突 
出 显示 PHP、HTML、CSS 和 SQL 语句 。 在 编写 代码 的 过 程 中 能 够 提供 函数 列表 参考 、 函 数 
参数 参考 ， 可 以 搜索 和 检测 编程 语法 等 。 总 之 ， 这 是 一 款 完 全 免费 的 优秀 PHP 编辑 器 。 


3. phpDesigner 


phpDesigner 是 一 款 功能 强大 的 、 运 行 高 效 的 、 优 秀 的 PHP 编辑 平台 。 它 是 结合 PHP、 
XHTML、JavaScript、CSS 等 技术 的 综合 Web 应 用 开发 平台 。 它 能 够 自动 捕获 代码 文件 中 的 
class、function、variables 等 编程 元 素 ， 并 加 以 整理 ， 在 编程 过 程 中 给 予 提示 。 除 此 以 外 ， 它 
还 兼容 了 各 种 流行 的 类 库 和 框架 ， 可 以 协同 工作 ， 如 JavaScript 的 jQuery 库 、YUI 库 、 
prototype 库 等 。 此 外 还 有 PHP 流行 的 zend 框架 、symfony 框架 、cakephp 框架 、yii 框架 等 。 
另外 ， 它 还 拥有 xdebug 工具 、svn 版 本 管理 等 工具 。 可 以 说 ，phpDesigner 是 独立 于 Eclipse 
之 外 的 ， 集 PHP 开发 需求 之 大 成 的 又 一 款 优秀 平台 。 


4. Zend Studio 


Zend Studio 是 由 Zend 科技 开发 的 一 个 针对 PHP 的 全 面 的 开发 平台 。 这 个 IDE 融合 了 
Zend Server 和 Zend 框架 ， 并 且 融 合 了 Eclipse 开发 环境 。Eclipse 是 最 早 用 于 Java 的 IDE 环 
境 。 但 是 由 于 其 优良 的 特性 和 对 PHP 的 支持 ， 它 已 经 成 为 很 有 影响 力 的 PHP 开发 工具 。 
Eclipse PHP 拥有 支持 Windows、Linux 和 Mac 系统 的 软件 包 ， 可 以 说 是 十 分 全 面 的 ， 拥 有 比 
较 完备 的 体系 ， 但 它 是 一 个 收费 的 工具 。 


1.3.2 ”网 页 设计 工具 


Dreamweaver 是 网 页 制作 “三 剑客 ”之 一 ， 其 功能 更 多 地 是 体现 在 对 Web 页 面 的 HTML 设 
计 上 。 随 着 Web 语言 的 发 展 ，Dreamweaver 早已 不 再 局 限于 网 页 设计 方面 ， 它 更 多 地 着 重 支 持 
各 种 流行 的 Web 前 后 台 技 术 的 综合 运用 。Dreamweaver CS6 对 PHP 的 支持 十 分 到 位 。 不 但 对 
PHP 的 不 同方 面 能 够 清晰 地 进行 表示 ， 并 且 能 够 给 出 足够 的 编程 提示 ， 使 编程 过 程 相当 流畅 。 


1.3.3 ”文本 编辑 工具 


常见 的 文本 编辑 工具 很 多 ， 如 UltraEdit 和 记事 本 等 。 
1. UltraEdit 


UltraEdit 是 一 款 功 能 强大 的 文本 编辑 器 ， 可 以 编辑 文本 、 十 六 进 制 码 、ASCI 码 ， 完 全 
可 以 取代 记事 本 (如 果 电 脑 配置 足够 强大 )。UltraEdit 内 建英 文 单字 检查 、C++ 及 VB 指令 突 
显 ， 可 同时 编辑 多 个 文件 ， 而 且 即 使 开启 很 大 的 文件 ， 速 度 也 不 会 慢 。 软 件 附 有 HTML 标签 
颜色 显示 、 搜 寻 蔡 换 以 及 无 限制 的 还 原 功 能 ， 一 般 用 来 修改 EXE 或 DLL 文件 ， 是 能 够 满足 
我 们 一 切 编辑 需要 的 编辑 器 。 

2. 记事 本 

记事 本 是 Windows 系统 自 带 的 文本 编辑 工具 。 它 具备 最 基本 的 文本 编辑 功能 ， 体 积 小 
巧 、 启 动 快 、 占 用 内 存 低 、 容 易 使 用 。 记 事 本 的 主 窗口 如 图 1-1 所 示 。 

局 无 标量 - 记事 本 - 0 x 


文件 (月 ”篇 硬 (E) 相 式 (0) 前 看 (V) 才 助 (H) 
<!DOCTYPE html> 





ee 《记事 本 也 能 编辑 PHP 文 档 ! “: 


</body> 
/html> 





图 1-1 记事 本 的 主 窗口 
在 使 用 记事 本 程序 编辑 PHP 文档 的 过 程 中 ， 需 要 注意 保存 方法 和 技巧 。 在 “另存 为 ”对 


话 框 中 输入 文件 名 称 ， 后 缀 名 为 .php， 同 时 “保存 类 型 ”设置 为 “所 有 文件 ” 即 可 ， 如 图 1-2 


所 示 。 
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1-2 “另存 为 ”对 话 框 


1.4 PHP7 的 新 特征 


PHP 7 是 PHP 编程 语言 的 一 个 主要 版 本 ， 是 开发 Web 应 用 程序 的 一 次 革命 ， 可 开发 和 交 
付 移动 企业 和 云 应 用 。 此 版 本 被 认为 是 PHP 5 后 最 重要 的 变化 。 
和 早期 版 本 相 比 ，PHP 7 有 以 下 几 个 新 的 特点 。 


1. 标量 类 型 声明 


PHP 7 增加 了 对 返回 类 型 声明 的 支持 。 返 回 类 型 声明 指明 了 函数 返回 值 的 类 型 。 可 用 的 
类 型 与 参数 声明 中 可 用 的 类 型 相同 。 例 如 以 下 代码 : 


<?php 

function arraysSuml(array ...$arrays): array 

{ 

return array map (function(array $array): int { 
return array_sum($array); 

}, $arrays); 

; 

print_r(arraysSsum([1,2,3], [4,5,6], [7,8,9])); 


> 

以 上 例子 会 输出 : 
Array 

( 

[0] => 6 

加 Js 5 

[2] => 24 


) 


2. null 合并 运算 符 
新 增 了 null 合并 运算 符 “? ? ”， 它 可 以 替换 三 元 表达 式 和 isset0。 例 如 以 下 代码 : 


$a = isset($ GET['a']) 2 $ GET['a'] : 1; 


可 以 用 null 合并 运算 符 蔡 换 如 下 : 


$a = $_ GET['a'] ?2 1; 


这 两 个 语句 的 含义 都 是 : 如 果 变 量 a 存在 且 值 不 为 NULL， 它 就 会 返回 自身 的 值 ， 否 则 
返回 它 的 第 二 个 操作 数 。 可 见 ， 新 增 的 ?? 运 算 符 可 以 简化 判断 语句 。 


3. 组 合 比 较 符 


组 合 比较 符 <=> 用 于 比较 两 个 表达 式 。 例 如 $a<=>$Sb， 表 示 当 $a 大 于 、 等 于 或 小 于 $b 时 
它 分 别 返回 1、0 或 -1。 例 如 以 下 代码 : 


<?php 

// 整 型 举例 

echo 1 <=> 1; // 输 出 0 
echo 1 <=> 2; // 输 出 -1 
echo 2 <=> 1; // 输出 1 

// 浮 点 型 举例 

echo 5.5 <=> 5.5 // 输 出 0 
echo 5.5 <=> 7.0; // 输 出 -1 
echo 7.0 <=> 5.5; // 输 出 1 
// 字 符 串 型 举例 

echo "a” <=> "a"; // 输 出 0 
echo "a" <=> "b"; // 输 出 -1 
echo "b"”<=> "a"; // 输 出 1 
?> 


4. 通过 define() 定 义 常量 数组 
对 于 常量 数组 ， 可 以 使 用 define0 定 义 。 例 如 以 下 代码 : 


<?php 

define('PERSON', ['xiaoming', 'xiaoli', 'xiaolan']); 
echo PERSON[1]; // 输出 "xiaoli" 

2 
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5. 匿名 类 


现在 支持 通过 new class 来 实例 化 一 个 匿名 类 ， 这 可 以 用 来 蔡 代 一 些 “ 用 后 即 焚 ”的 完整 
类 定义 。 

6. 支持 Unicode 字符 格式 

PHP 7 支持 任何 有 效 的 codepoint 编码 ， 输 出 为 UTF-8 编码 格式 的 字符 串 。 例 如 以 下 代码 : 


<?php 

echo "\u{6666}"; 

2> 

在 PHP 7 环境 下 输出 为 : 星 ， 而 在 早期 版 本 中 则 输出 为 : \n{6666} 
7. 更 多 的 Error 变 为 可 捕获 的 Exception 


PHP 7 实现 了 一 个 全 局 的 throwable 接口 ， 原 来 的 Exception 和 部 分 Error 都 实现 了 这 个 接 
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口 (interface)， 以 接口 的 方式 定义 了 异常 的 继承 结构 。 于 是 ，PHP 7 中 更 多 的 Error 变 为 可 捕获 
的 Exception 返回 给 开发 者 。 如 果 不 进行 捕获 则 为 Error; 如 果 捕 获 就 变 为 一 个 可 在 程序 内 处 
理 的 Exception。 这 些 可 被 捕获 的 Error 通常 都 是 不 会 对 程序 造成 致命 伤害 的 Error， 如 函数 不 
存在 。PHP 7 进一步 方便 开发 者 处 理 ， 让 开发 者 对 程序 的 掌控 能 力 更 强 。 因 为 在 默认 情况 
下 ，Error 会 直接 导致 程序 中 断 ， 而 PHP 7 则 提供 捕获 并 且 处 理 的 能 力 ， 让 程序 继续 执行 下 
去 ， 为 程序 员 提供 更 灵活 的 选择 。 

例如 ， 执 行 一 个 不 确定 是 否 存在 的 函数 ，PHP 5 兼容 的 做 法 是 在 函数 被 调用 之 前 追加 的 
判断 function exist， 而 PHP 7 则 支持 捕获 Exception 的 处 理 方式 。 


8. 性 能 大 幅度 提升 


PHP 7 与 PHP 5 相 比 ， 速 度 快 2 倍 以 上 。 另 外 ，PHP 7 降低 了 内 存 消耗 。 优 化 后 PHP 7 
使 用 较 少 的 资源 ， 比 PHP 5.6 低 了 50% 的 内 存 消 耗 。 同 时 ，PHP 7 也 支持 64 位 架构 的 机 器 ， 
运算 速度 更 快 。PHP 7 可 以 服务 于 更 多 的 并 发 用 户 ， 无 须 任何 额外 的 硬件 。 


1.5 “如何 能 学 好 PHP 7 


对 初学 者 而 言 ， 如 何 能 快速 学 好 PHP 7， 然 后 开发 出 功能 齐全 的 动态 网 站 ， 才 是 大 家 最 
关心 的 问题 。 根 据 多 年 的 教学 经 验 ， 为 初学 者 规划 了 一 个 学 习 路 线 图 ， 如 图 1-3 所 示 。 


步骤 1 ; 快速 熟悉 HTNL5、CSS3 和 JavaScript 等 
网 页 基本 元 素 ， 可 以 自行 设计 一 个 简单 的 网 页 。 


步骤 2， 理 解 动态 语言 的 概念 和 运行 机 制 ， 掌 握 
基本 的 PHP 语 法 。 


和 将 PHP 与 HTIL 5 结合 起 来 完成 简单 的 动态 


| 步 慑 4 训 斩 yS 掌握 PHP 是 如 何 
虞 作 hMysQL 数 据 库 的 


步骤 5， 不 断 巩固 PHP 语 法 ， 熟悉 大 部 分 的 PHP 常 用 及 雪 
理解 面向 对 象 编程 、MySQL 优 化 ， 以 及 一 些 模板 和 框架 。 





了 睹 驯 5: 挑选 一 Ee 最 终 能 自己 独立 开发 
个 功能 天 全 的 动态 网 站 


图 1-3 学 习 路 线 图 
学 习 PHP 编程 语言 ， 一 定 要 反复 地 思考 和 不 断 地 实践 。 在 学 习 PHP 语言 之 前 ， 首 先 要 熟 
悉 HIML 5、CSS 3 和 JavaScript 等 基本 技术 ， 从 而 可 以 制作 一 个 静态 的 网 页 ， 这 是 开发 动态 


网 站 前 的 基本 要 求 。 其 中 HTML 5 中 常用 元 素 可 以 通过 亲自 实践 的 方式 快速 理解 。CSS 3 是 
网 页 优化 中 比较 好 的 搭配 技术 ， 需 要 读者 深刻 理解 ， 勤 加 练习 。 


当 读 者 可 以 独立 完成 一 个 漂亮 的 静态 页 面 后 ， 就 可 以 进一步 了 解 动态 语言 的 运行 原理 和 
PHP 的 基本 语法 。 这 里 一 定 要 明白 静态 网 页 和 动态 网 页 的 不 同 之 处 ， 同 时 理解 动态 语言 是 经 
过 PHP 解析 器 后 输出 到 浏览 器 的 。 

了 解 了 PHP 的 基本 语法 ， 能 做 出 一 些 简单 的 动态 网 页 后 ， 读 者 即 可 开始 学 习 MySQL 数 
据 库 和 PHP 操作 MySQL 数据 库 的 方法 和 技能 。PHP+MySQL 是 开发 动态 网 页 的 黄金 措 档 ， 
所 以 一 定 要 掌握 这 些 基 本 技能 。 

接 下 来 就 可 以 不 断 巩固 所 学 的 知识 ， 学 习 PHP 的 高 级 技能 ， 包 括 常 用 函数 、 面 向 对 象 编 
程 、 模 板 和 框架 等 不 容易 理解 的 知识 。PHP 框架 是 一 个 可 以 用 来 节省 时 间 并 强化 自己 代码 的 工 
具 。 对 于 常见 的 框架 ， 读 者 要 进行 对 比 学 习 ， 分 析 它 们 的 优 缺 点 ， 以 便于 后 期 的 项 目 制作 。 

如 果 前 面 的 学 习 比 较 顺 利 ， 就 可 以 尝试 做 一 些 经 典 PHP 项 目 ， 除 了 提升 技能 和 积累 项 目 
经 验 ， 还 可 以 了 解 项 目 开 发 的 思路 和 技巧 。 

最 后 由 囊 地 提醒 大 家 ， 学 习 PHP 技术 可 能 会 遇 到 一 些 困 难 。 读 者 有 时 候 会 比较 迷茫 ， 这 
时 候 需要 持之以恒 的 精神 ， 多 实践 、 多 积累 ， 才 能 把 PHP 技术 学 好 。 


1.6 疑难 解 惑 


疑问 1: 如 何 快速 了 解 PHP 的 应 用 技术 ? 


答 : 在 学 习 的 过 程 中 ， 用 户 可 以 随时 查阅 PHP 的 相关 资料 。 启 动 正 浏览 器 ， 在 地 址 栏 中 
输入 “http://www.baidu.com”， 打 开 搜 索引 擎 ， 输 入 需要 搜索 的 内 容 ， 即 可 了 解 相 关 的 技术 。 


疑问 2: 如何 选 择 PHP 开发 软件 ? 


答 : 不 管 是 哪个 开发 工具 ， 在 PHP 开发 过 程 中 ， 都 要 有 对 PHP 的 语法 和 数据 进行 分 色 表 
示 的 能 力 ， 以 方便 开发 者 编写 程序 。 进 一 步 的 功能 是 对 代码 编写 要 拥有 提示 的 能 力 ， 即 对 
PHP 的 数据 类 型 、 运 算 符 、 标 识 、 名 称 等 都 能 给 出 提示 。 

在 诸多 开发 工具 中 ， 我 们 选择 比较 适合 自己 的 即 可 。 就 初学 者 而 言 ， 使 用 phpDesigner 比 
较 好 ， 它 支持 PHP、XHTML、JavaScript、CSS 等 Web 开发 的 常用 技术 。 
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第 2 章 
开发 前 必 备 工作 
一 配置 PHP 

服务 器 环境 


在 编写 PHP 文件 之 前 ， 用 户 需 要 配置 PHP 服务 器 ， 包 括 软 硬 件 环境 的 检查 ， 
如 何 获得 PHP 安装 资源 包 等 。 本 章 详细 讲解 目前 常见 的 主流 PHP 服务 器 搭配 方 
案 : PHP+IIS 和 PHP+Apache。 然 后 讲述 了 在 Windows 下 如 何 使 用 WampServer 组 
合 包 。 最 后 通过 一 个 测试 案例 ， 可 以 检查 Web 服务 器 建构 是 否 成 功 。 
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2.1 PHP 服务 器 概述 


在 学 习 PHP 服务 器 之 前 ， 需 要 了 解 HTML 网 页 的 运行 原理 。 网 页 浏览 者 在 客户 端 通过 浏 
览 器 向 服务 器 发 出 页 面 请 求 ， 服 务 器 接收 到 请 求 后 将 页 面 返回 到 客户 端的 浏览 器 ， 这 样 网 页 
浏览 者 即 可 看 到 页 面 显示 效果 。 

PHP 语言 在 Web 开发 中 作为 嵌入 式 语言 ， 需 要 嵌入 到 HTML 代码 中 执行 。 要 想 运 行 
PHP 网 站 ， 需 要 搭建 PHP 服务 器 。PHP 网 站 的 运行 原理 如 图 2-1 所 示 。 





2-1 PHP 网 站 的 运行 原理 


从 图 2-1 可 以 看 出 ，PHP 程序 运行 的 基本 流程 如 下 。 

(1) 网 页 浏览 者 首先 在 浏览 器 的 地 址 栏 中 输入 要 访问 的 主页 地 址 ， 按 Enter 键 触发 该 请 求 。 
(2) 浏览 器 将 请 求 发 送 到 PHP 网 站 服务 器 。 网 站 服务 器 根据 请 求 读 取 数据 库 中 的 页 面 。 
(3) 通过 Web 服务 器 向 客户 端 发 送 处 理 结果 ， 客 户 端的 浏览 器 显示 最 终 页 面 。 


Se 由 于 在 客户 端 显示 的 只 是 服务 器 端 处 理 过 的 HTML 代码 页 面 ， 所 以 网 页 浏览 者 
SS 看 不 到 PHP 代码 ， 这 样 可 以 提高 代码 的 安全 性 。 同 时 ， 在 客户 端 不 需要 配置 PHP 环 


境 ， 只 要 安装 浏览 器 即 可 。 


2.2 安装 PHP 前 的 准备 工作 
在 安装 PHP 之 前 ， 需 要 了 解 安 装 所 需要 的 软 硬 件 环境 和 获取 PHP 安装 资源 包 的 途径 。 


2.2.1 软 硬 件 环 境 


大 部 分 软件 在 安装 的 过 程 中 都 需要 软 硬 件 环境 的 支持 ， 当 然 PHP 也 不 例外 。 在 硬件 方 
面 ， 如 果 只 是 为 了 学 习 上 的 需求 ，PHP 只 需要 一 台 普通 的 计算 机 即 可 。 在 软件 方面 ， 需 要 根 
据 实际 工作 的 需求 选择 不 同 的 Web 服务 器 软件 。 

PHP 具有 跨 平台 特性 ， 所 以 PHP 开发 用 什么 样 的 系统 不 太 重 要 ， 开 发 出 来 的 程序 能 够 很 
轻松 地 移植 到 其 他 操作 系统 中 。 另 外 ，PHP 开发 平台 支持 目前 主流 的 操作 系统 ， 包 括 
Windows、Linux、UNIX 和 Mac OS X 等 。 本 书 以 Windows 平台 为 例 进行 讲解 。 

另外 ， 用 户 还 需要 安装 Web 服务 器 软件 。 目 前 ，PHP 支持 大 多 数 Web 服务 器 软件 ， 常 
见 的 有 IS、Apache、PWS 等 。 比 较 流行 的 是 IS 和 Apache， 下 面 详细 讲述 这 两 种 Web 服务 
器 的 安装 和 配置 方法 。 


2.2.2 实例 1 一 一 获取 PHP 7.1 安装 资源 包 


PHP 安装 资源 包 中 包括 了 安装 和 配置 PHP 服务 器 的 所 需 文件 和 PHP 扩展 函数 库 。 获 取 
PHP 安装 资源 包 的 方法 比较 多 。 很 多 网 站 都 提供 PHP 安装 包 ， 但 是 建议 读者 从 官方 网 站 下 
载 。 具 体操 作 步 骤 如 下 。 

EEC 打开 下 浏览 器 ， 在 地 址 栏 中 输入 下 载 地 址 “http://windows.php.net/download”， 

按 Enter 键 确认 ， 登 录 到 PHP 下 载 网 站 ， 如 图 2-2 所 示 。 
ED 进入 下 载 页 面 ， 单 击 Binaries and sources Releases 下 拉 列 表 框 右 侧 的 下 三 角 按 
钮 ， 在 打开 的 下 拉 列 表 中 选择 合适 的 版 本 ， 这 里 选择 PHP 7.1 版 本 ， 如 图 2-3 所 示 。 
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图 2-2 PHP 网 站 下 载 页 面 图 2-3 选择 需要 的 版 本 
Se 在 图 2-3 中 ， 下 拉 列 表 中 VC11 代表 的 是 the Visual Studio 2012 compiler 编译 器 编 
站 译 ， 通 常用 于 PHP+IIS 服务 器 下 。 要 求 用 户 安装 Visual C++ Redistributable for Visual 
Studio 2012。 


ETEY 页 面 中 显示 所 选 版 本 号 中 PHP 安装 包 的 各 种 格式 。 这 里 选择 Zip 压缩 格式 ， 单 
击 Zip 链接 ， 如 图 2-4 所 示 。 
EEC 打开 “另存 为 ”对 话 框 ， 选 择 保存 路 径 ， 然 后 保存 文件 即 可 ， 如 图 2-5 所 示 。 
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图 2-4 选择 需要 版 本 的 格式 图 2-5 “另存 为 ”对 话 框 
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2.3 PHP +llS 服务 器 的 安装 配置 


下 面 介 绍 PHP +IIS 服务 器 架构 的 配置 方法 和 技巧 。 


2.3.1 实例 2 一 一 llS 简介 及 安装 


IIS 是 Internet Information Services( 互 联网 信息 服务 ) 的 简称 ， 是 微软 公司 提供 的 基于 
Microsoft Windows 的 互联 网 基本 服务 。 由 于 功能 强大 、 操 作 简 单 和 使 用 方便 ， 所 以 它 是 目前 
较为 流行 的 Web 服务 器 之 一 。 

目前 IS 只 能 运行 在 Windows 系列 的 操作 系统 上 。 针 对 不 同 的 操作 系统 ，IIS 也 有 不 同 的 
版 本 。 下 面 以 Windows 10 为 例 进 行 讲解 ， 在 默认 情况 下 此 操作 系统 没有 安装 IS 。 

安装 IIS 组 件 的 具体 操作 步骤 如 下 。 

EEIDp 右 击 “开始 ”按钮 ， 在 弹出 的 快捷 菜单 中 选择 “控制 面板 ”命令 ， 如 图 2-6 

所 示 。 
EDp 打开 “控制 面板 ”窗口 ， 双 击 “ 程 序 ” 选 项 ， 如 图 2-7 所 示 。 
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闻 = 
图 2-6 选择 “控制 面板 ”命令 图 2-7 “控制 面板 ”窗口 
EEIS 打开 “程序 ”窗口 ， 从 中 单 击 “ 启 用 或 关闭 Windows 功能 ”文字 链接 ， 如 图 2-8 


所 示 。 
EC 在 打开 的 “Windows 功能 ”对 话 框 中 ， 选 中 “Intemet Information Services” 复 
选 框 ， 如 图 2-9 所 示 。 然 后 单 击 “确定 ”按钮 ， 开 始 安装 。 
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图 2-8 “程序 ”窗口 图 2-9 “Windows 功能 ”对 话 框 


ED 安装 完成 后 ， 即 可 测试 是 否 成 功 。 在 正 浏览 器 的 地 址 栏 中 输入 “http://localhost/”， 
打开 IIS 的 欢迎 页 面 ， 如 图 2-10 所 示 。 
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图 2-10 IIS 的 欢迎 页 面 


2.3.2 ”实例 3 一 一 PHP 的 安装 


IIS 安装 完成 后 ， 即 可 开始 安装 PHP。PHP 的 安装 过 程 大 致 分 成 3 个 步骤 。 
1. 解压 和 设置 安装 路 径 


将 获取 到 的 安装 资源 包 解压 缩 ， 解 压缩 后 得 到 的 文件 夹 中 存放 着 PHP 所 需要 的 文件 。 将 
文件 夹 复 制 到 PHP 的 安装 目录 中 。PHP 的 安装 路 径 可 以 根据 需要 进行 设置 。 例 如 ， 本 书 设置 
为 D\PHP7\， 文 件 夹 复制 后 的 效果 如 图 2-11 所 示 。 
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2-11 PHP 的 安装 目录 
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2. 配置 PHP 


在 安装 目录 中 ， 找 到 php.ini-development 文件 ， 此 文件 正 是 PHP 7.1 的 配置 文件 。 将 这 个 
文件 的 扩展 名 .ini-development 修改 为 .ini， 然 后 用 记事 本 打开 。 文 件 中 参数 很 多 ， 所 以 建议 读 
者 使 用 记事 本 的 查找 功能 ， 快 速 查 找 需要 的 参数 。 

查找 并 修改 相应 的 参数 值 ，extension_dir="D:\PHP7\ext"， 此 参数 为 PHP 扩展 函数 的 查找 
路 径 ， 其 中 D:\PHP7\ 为 PHP 的 安装 路 径 。 读 者 可 以 根据 自己 的 安装 路 径 进行 修改 。 采 用 同样 
的 方法 ， 修 改 参数 : cgi.force_redirect =0。 

另外 ， 去 除 下 面 的 参数 值 扩展 前 的 分 号 ， 最 终结 果 如 图 2-12 所 示 。 


?extension=php_bz2.d1l 
?iextension=php_curl.dll 
?extension=php_fileinfo.dll 
?extension=php_gd2.d1l 
?extension=php_gettext .dll 
?extension=php_gmp.d1l 
?extension=php_intl.dll 
?extension=php_imap.dll 
?extension=php_interbase.dll 
?extension=php_ldap.d1ll 
?extension=php mbstring.dll 
?extension=php_exif.dll 
?extension=php_mysqli.dll 
?extension=php_oci8_l2c.dll 
?extension=php_openssl.dll 
?extension=php_pdo_firebird.dll 
?extension=php_pdo_mysql .dll 
?extension=php_pdo_oci.dll 
?extension=php_pdo_odbc.d1ll 
?extension=php_pdo_pgsql .dll 
?extension=php_pdo_sqlite.dll 
?extension=php_pgsql.d1ll 
?extension=php_shmop.dll 
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rtension=php mysqli dil 
ztensicnrphp cci8_12c.dll ; Use with Oracle Database 12c Instant Clisnt 
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2-12 ”去 除 分 号 
3. 添加 系统 变量 
要 想 让 系统 运行 PHP 时 找到 上 面 的 安装 路 径 ， 就 需要 将 PHP 的 安装 目录 添加 到 系统 变量 


中 。 具 体操 作 步 又 如 下 。 
EC 右 击 桌面 上 的 “此 电脑 ”图 标 ， 在 弹出 的 快捷 菜单 中 选择 “属性 ”命令 ， 打 开 
“系统 ”窗口 ， 如 图 2-13 所 示 。 
EPY 单 击 “ 高 级 系统 设置 ”链接 ， 打 开 “ 系 统 属性 ”对 话 框 ， 如 图 2-14 所 示 。 
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图 2-13 “系统 ”窗口 


EEUS 在 “高 级 ”选项 卡 中 单 击 “ 环 境 变量 ”按钮 ， 打 开 “ 环 境 变量 ”对 话 框 。 在 
“系统 变量 ”列表 框 中 选择 变量 Path， 然 后 单 击 “编辑 ”按钮 ， 如 图 2-15 所 示 。 
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全 本 人- 
Ra | 
请 二 LL- 0 
图 2-14 “系统 属性 ”对 话 框 图 2-15 “环境 变量 ”对 话 框 
EC 弹出 “编辑 系统 变量 ”对 话 框 ， 在 “变量 值 ”文本 框 的 末尾 输入 “;d:\PHP7”， 
如 图 2-16 所 示 。 
注 辣 至 统 六 呈 六 
变量 和 (NI: path 
Vy In Flee\nte\WiFi\binU CDregram Files\Commen Files\InteN\V/irelessCommonud\PHD 
误区 目录 Di- 文件 站- E39 





图 2-16 “编辑 系统 变量 ”对 话 框 
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攻 BY 单 击 “ 确 定 ”按钮 ， 返 回 到 “环境 变量 ”对 话 框 ， 依 次 单 击 “ 确 定 ” 按 钮 即 可 
关闭 对 话 框 ， 然 后 重新 启动 计算 机 。 这 样 设置 的 环境 变量 即 可 生效 。 


2.3.3 ”实例 4 一 一 设置 虚拟 目录 


如 果 用 户 是 按照 前 述 的 方式 来 启动 IS 网 站 服务 器 的 ， 那 么 整个 网 站 服务 器 的 根 目录 就 
位 于 “(系统 盘 符 ):\Inetpub\wwwwroot” 中 ， 也 就 是 如 果 要 添加 网 页 到 网 站 中 显示 ， 都 必须 放置 
在 这 个 目录 之 下 。 但 是 会 发 现 这 个 路 径 不 仅 太 长 ， 也 不 好 记 ， 使 用 起 来 相当 不 方便 。 

这 些 问 题 都 可 以 通过 修改 虚拟 目录 来 解决 。 具 体操 作 步 骤 如 下 。 

EEC 在 桌面 上 右 击 “此 电脑 ”图 标 ， 在 弹出 的 快捷 菜单 中 选择 “管理 ”命令 ， 打 开 

“计算 机 管理 ”窗口 ， 在 左 侧 的 列表 框 中 展开 “服务 和 应 用 程序 ”选项 ， 选 择 
“Internet 信息 服务 (IIS) 管 理 器 ”选项 ， 在 右 侧 选择 Default Web Site 选项 后 ， 右 击 
并 在 弹出 的 快捷 菜单 中 选择 “添加 虚拟 目录 ”命令 ， 如 图 2-17 所 示 。 
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图 2-17 “计算 机 管理 ”窗口 
EEJDp 弹出 “添加 虚拟 目录 ”对 话 框 ， 在 “ 别 a 
名 ”文本 框 中 输入 虚拟 网 站 的 名 称 ， 这 里 输入 ne 
“php7”， 然 后 设置 物理 路 径 为 D:\php( 该 NE/ 
文件 夹 必须 已 存在 )， 单 击 “ 确 定 ” 按 钮 ， 如 [ 
图 2-18 所 示 。 5 内 
至 此 ， 已 完成 了 JS 网 站 服务 器 设置 的 更 改 。 IIS | Be 品 
网 站 服务 器 的 网 站 虚拟 目录 已 经 更 改 为 “D:\php” 了 。 i 




















图 2-18 “添加 虚拟 目录 ”对 话 框 


2.4 PHP+Apache 服务 器 的 环境 搭建 


Apache 支持 大 部 分 操作 系统 ， 搭 配 PHP 程序 的 应 用 ， 即 可 开发 出 功能 强大 的 互动 网 站 。 
下 面 主要 讲述 PHP+Apache 服务 器 的 搭建 方法 。 

















2.4.1 Apache 简介 


Apache 可 以 运行 在 几乎 所 有 的 计算 机 平台 上 ， 由 于 其 跨 平台 和 安全 性 而 被 广泛 使 用 ， 是 
目前 最 流行 的 Web 服务 器 端 软 件 之 一 。 

和 一 般 的 Web 服务 器 相 比 ，Apache 的 主要 特点 如 下 。 

(1) 跨 平 台 应 用 。 几 乎 可 以 在 所 有 的 计算 机 平台 上 运行 。 

(2) 开放 源 代 码 。Apache 服务 程序 由 全 世界 的 众多 开发 者 共同 维护 ， 并 且 任 何人 都 可 以 
自由 使 用 ， 充 分 体现 了 开源 软件 的 精神 。 

(3) 支持 HITP 1.1 协议 。Apache 是 最 先 使 用 HITP 1.1 协议 的 Web 服务 器 之 一 ， 它 完 
兼容 HTTP 1.1 协议 并 与 HTTP 1.0 协议 向 后 兼容 。Apache 已 为 新 协议 所 提供 的 全 部 内 容 做 好 
了 必要 的 准备 。 

(4) 支持 通用 网 关 接口 CCGD。Apache 遵守 CGI 1.1 标准 并 且 提 供 了 扩充 的 特征 ， 如 定制 
环境 变量 和 很 难 在 其 他 Web 服务 器 中 找到 的 调试 支持 功能 。 

(5) 支持 常见 的 网 页 编程 语言 。 可 支持 的 网 页 编程 语言 包括 Perl、PHP、Python、Java 
等 。 支 持 各 种 常用 的 Web 编程 语言 使 Apache 具有 更 广泛 的 应 用 领域 。 

(6) 模块 化 设计 。 通 过 标准 的 模块 实现 专 有 的 功能 ， 提 高 了 项 目 完成 的 效率 。 

(7) 运行 非常 稳定 ， 同 时 具备 效率 高 、 成 本 低 的 特点 ， 而 且 具 有 良好 的 安全 性 。 


2.4.2 ”实例 5 一 一 关闭 原 有 的 网 站 服务 器 


在 安装 Apache 网 站 服务 器 之 前 ， 如 果 所 使 用 的 操作 系统 已 经 安装 了 网 站 服务 器 ， 如 IIS 
网 站 服务 器 等 ， 用 户 必须 先 停止 这 些 服 务 器 ， 才 能 正确 安装 Apache 网 站 服务 器 。 

以 Windows 10 操作 系统 为 例 ， 在 桌面 上 右 击 “ 此 电脑 ”图 标 ， 在 弹出 的 快捷 菜单 中 选 
择 “ 管 理 ” 命 令 ， 打 开 “ 计 算 机 管理 ”窗口 ， 在 左 侧 的 列表 框 中 展开 “服务 和 应 用 程序 ” 选 
项 ， 然 后 选择 Intemet Information 选项 ， 在 右 侧 的 列表 框 中 单 击 “停止 ”链接 即 可 停止 HS 服 
务 器 ， 如 图 2-19 所 示 。 

如 此 一 来 ， 原 来 的 服务 器 软件 即 失效 ， 不 再 工作 ， 也 不 会 与 即将 安装 的 Apache 网 站 服 
务 器 产生 冲突 。 当 然 ， 如 果 用 户 的 系统 原来 就 没有 安装 IIS 等 服务 器 软件 ， 则 可 略 过 这 一 小 
节 的 步骤 直接 进行 服务 器 的 安装 。 
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2-19 停止 IIS 服务 器 


2.4.3 ”实例 6 一 一 安装 Apache 
Apache 是 免费 软件 ， 用 户 可 以 从 官方 网 站 直接 下 载 。Apache 的 官方 网 站 为 http://www. 


apache.org。 
下 面 以 Apache 2.2 为 例 ， 讲 解 如 何 安装 Apache。 有 具体 操作 步骤 如 下 。 
EEC 双击 Apache 安装 程序 ， 打 开 安 装 向 导 欢 迎 界面 ， 单 击 Next 按钮 ， 如 图 2-20 
所 示 。 
EECJDp 弹出 Apache 许可 协议 界面 ， 阅 读 完 后 ， 选 中 I accept the terms in the license 
agreement 单 选 按钮 ， 单 击 Next 按钮 ， 如 图 2-21 所 示 。 





请 Apadhe HTTP Sewer 22- Instaladon Wizard x WA 
been Agreement 
Welcome to the nstallation Wlzard for 
Apache HTTP Server 2.2.25 Please read the folowng hcenee a0reement carefly 
The Daialaton Wuard wl nstal Mpache HTP server 22.25 Apache License 加 
on Your computer, To artnue cek Next. Version 2.0, January 2004 
hap:wwwapache orglicenses/ 


WS AND CONDITIDNS FOR USE REPRODUCTION, AND DISTRIBUTION 
| Definitions. 
WARNING: Ths program = protacied by capyrght aw and "License” shall mean the terms and condilons for use reproducion, and 
‘rhea etioe, istriouiion as definad by Secions 1 hrouoh 9 of this document 
©: accept the terme in the icenee agreement 
Did rotaceept the temsin the license ageement 


| Irstalsried 


























Ee ] | ee P| 7 Co 
图 2-20 欢迎 界面 图 2-21 Apache 许可 协议 界面 
ES 弹出 Apache 服务 器 注意 事项 界面 ， 阅 读 完毕 后 ， 单 击 Next 按钮 ， 如 图 2-22 


所 示 。 

EC 弹出 服务 器 信息 设置 界面 ， 输 入 服务 器 的 一 些 基本 信息 ， 分 别 为 Network 
Domain( 网 络 域名 )、Server Name( 服 务 器 名 )、Administrator's Email Address( 管 理 员 信 
箱 ) 和 Apache 的 工作 方式 。 如 果 只 是 在 本 地 计算 机 上 使 用 Apache， 前 两 项 可 以 输入 

“localhost”。 工 作 方式 建议 选择 第 一 项 : 针对 所 有 用 户 ， 工 作 端口 为 80， 当 机 器 


启动 时 自动 启动 Apache。 单 击 Next 按钮 ， 
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2-22 Apache 服务 器 注意 事项 界面 


如 图 2-23 所 示 。 
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2-23 ”服务 器 信息 设置 界面 


国 于 BY 弹出 安装 类 型 界面 ， 其 中 Typical 为 典型 安装 ，Custom 为 自 定义 安装 。 通 常情 
况 下 ， 选 择 典型 安装 即 可 ， 单 击 Next 按钮 ， 如 图 2-24 所 示 。 

EEJD 弹出 安装 路 征 选 择 界面 ， 单 击 Change 按钮 ， 可 以 重新 设置 安装 路 径 ， 本 实例 采 
用 默认 安装 路 径 ， 单 击 Next 按钮 ， 如 图 2-25 所 示 。 


人 名 Apache HTTP Server 2.2 - Installation Wizard 
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Choose the setup tyoe that best ita your reeds, 
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图 2-24 安装 类 型 界面 


2-25 ”安装 路 径 选 择 界面 


ET 弹出 安装 准备 就 绪 界 面 ， 单 击 Install 按钮 ， 如 图 2-26 所 示 。 
GEE> 系统 开始 自动 安装 Apache 主 程序 ， 安 装 完成 后 ， 弹 出 提示 信息 界面 ， 单 击 


Finish 按钮 关闭 对 话 框 ， 如 图 2-27 所 示 。 
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2-26 ”安装 准备 就 绪 界 面 


2-27 Apache 安装 完成 
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2.4.4 ”实例 7 一 一 将 PHP 与 Apache 建立 关联 


Apache 安装 完成 后 ， 还 不 能 运行 PHP 网 页 ， 需 要 将 PHP 与 Apache 建立 关联 。 

Apache 的 配置 文件 名 称 为 httpd.conf， 此 为 纯 文本 文件 ， 用 记事 本 即 可 打开 编辑 。 此 文件 
存放 在 Apache 安装 目录 的 Apache2vconfig\ 目 录 下 。 另 外 ， 也 可 以 通过 单 击 “ 开 始 ”按钮 ， 在 
打开 的 菜单 中 选择 Apache HTTP Server 2.2 一 Edit the Apache httpd.conf Configuration File 命 
令 ， 如 图 2-28 所 示 。 

打开 Apache 的 配置 文件 后 ， 首 先 设置 网 站 的 主 目录 。 例 如 : 如 果 将 案例 的 源 文件 放 在 D 
盘 的 php7book 文件 夹 下 ， 则 主 目 录 就 需要 设置 为 d:/php7book/。 在 httpd.conf 配置 文件 中 找到 
DocumentRoot 参数 ， 将 其 值 修改 为 "d:/php7book/"， 如 图 2-29 所 示 。 











ff which you will serve your 
uests are taken fron this directory, but 
alinses xay be usod to point to cther locations, 











图 2-28 选择 Apache 配置 文件 图 2-29 设置 网 站 的 主 目录 


下 面 指定 php.ini 文件 的 存放 位 置 。 由 于 PHP 安装 在 dphp7， 所 以 php.ini 位 于 
d:\php7\php.ini。 在 httpd.conf 配置 文件 中 的 任意 位 置 输入 语句 “PHPIniDir"d:\php7\php.ini"”， 
如 图 2-30 所 示 。 

最 后 向 Apache 中 加 入 PHP 模块 。 在 httpd.conf 配置 文件 中 的 任意 位 置 加 入 3 行 语句 : 


LoadModule php7 module"d:/php7/php7apache2 2.d11" 

AddType application/x-httpd-php .php 

AddType application/x-httpd-php .html 

输入 效果 如 图 2-31 所 示 。 完 成 上 述 操作 后 ， 保 存 httpd.conf 文件 即 可 。 然 后 重启 Apache， 
即 可 使 设置 生效 。 
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2-30 指定 php.ini 文件 的 存放 位 置 2-31 向 Apache 中 加 入 PHP 模块 


2.5 新手 的 福音 一 一 安装 WampServer 集成 开发 环境 


对 于 刚 开始 学 习 PHP 的 程序 员 ， 往 往 因为 配置 环境 而 不 知 所 措 ， 为 此 本 节 讲 述 
WampServer 组 合 包 的 使 用 方法 。WampServer 组 合 包 是 将 Apache、PHP、MYySQL 等 服务 器 软 
件 安装 配置 完成 后 打包 处 理 。 因 为 它 安装 简单 、 速 度 较 快 、 运 行 稳定 ， 所 以 受到 广大 初学 者 
的 青睐 。 

Sn 在 安装 WampServer 组 合 包 之 前 ， 需 要 确保 系统 中 没有 安装 Apache、PHP 和 

站 MySQL。 否则， 需要 先 将 这 些 软件 趣 载 ， 然 后 才能 安装 WampServer 组 合 包 。 


安装 WampServer 组 合 包 的 具体 操作 步骤 如 下 。 

到 WampServer 官方 网 站 http://www.wampserver.com/en/ 下 载 WampServer 的 最 
新 安装 包 WampServer3.0.6-x32.exe 文件 。 

EEJPy 直接 双击 安装 文件 ， 打 开 选 择 安装 语言 界面 ， 如 图 2-32 所 示 。 

单 击 OK 按钮 ， 在 弹出 的 对 话 框 中 选中 I accept the agreement 单 选 按钮 ， 如 图 2-33 
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所 示 。 
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ee 
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ES | 本 
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图 2-32 选择 安装 语言 界面 图 2-33 ”接受 许可 证 协议 


ZY 单 击 Next 按钮 ， 弹 出 Information 界面 ， 在 其 中 可 以 查看 组 合 包 的 相关 说 明 信 
息 ， 如 图 2-34 所 示 。 

EECJD) 单 击 Next 按钮 ， 在 弹出 的 界面 中 设置 安装 路 径 ， 这 里 采用 默认 路 径 cvwamp， 
如 图 2-35 所 示 。 

ECED) 单 击 Next 按钮 ， 在 弹出 的 界面 中 选择 开始 菜单 文件 夹 ， 这 里 采用 默认 设置 ， 
如 图 2-36 所 示 。 

EC 单 击 Next 按钮 ， 在 弹出 的 界面 中 确认 安装 的 参数 后 ， 单 击 Install 按钮 ， 如 
图 2-37 所 示 。 

TBY 程序 开始 自动 安装 ， 并 显示 安装 进度 ， 如 图 2-38 所 示 。 

安装 完成 后 ， 进 入 安装 完成 界面 ， 单 击 Finish 按钮 ， 完 成 WampServer 的 安装 操 
作 ， 如 图 2-39 所 示 。 
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[-- Installation of Wampserver --- 

[eroRE procseding weh the nctallation of Wampsever, you mudt| 
ensure that ceran elemens are installed on your system, othewtse 
|wampserver will aosolutsly not run. and in addition. the installation will 
be toulty and you nesd to remove Wampserver 3EFORE inetaling the 
lements hat were msng 

|Make sure you are "up to date” in the redistributable packages 








es vowe, ven, ve ond ve™s 
see -— VBUa/ C++ Packages below. 
~ Do not install Wampserver OVER an existing version, folow the 


[ae 





mal ew veglon of 
(itsoumyanoserer com/read op'2 123606 


| ood re ] emm | 





Wanpserver Tretall by Tnne san 





加 Sehp - Wampsever2 


Select Destination Loation @ 


Where shoud Wanpserver32 be nstaled? 
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| 图 2-34 信息 界面 
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图 2-36 设置 开始 菜单 文件 夹 
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2-35 设置 安装 路 径 
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图 2-37 确认 安装 
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2-38 ”正在 安装 程序 


2-39 ”完成 安装 界面 


在 默认 情况 下 ， 集 成 环境 中 的 PHP 版 本 为 5.6.25， 这 里 需要 修改 为 最 新 的 PHP 7 版 
本 。 单 击 桌 面 右 侧 的 WampServer 服务 按钮 国 ， 在 弹出 的 下 拉 菜单 中 选择 PHP 命令 ， 然 
后 在 弹出 的 子 菜单 中 选择 Version 命令 ， 选 择 PHP 的 版 本 为 7.0.10， 如 图 2-40 所 示 。 

单 击 桌面 右 侧 的 WampServer 服务 按钮 国 ， 在 弹出 的 下 拉 菜 单 中 选择 Localhost 
命令 ， 如 图 2-41 所 示 。 
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2-40 WampServer 服务 列表 图 2-41 选择 Localhost 命令 
ED9 系统 自动 打开 浏览 器 ， 显 示 PHP 配置 环境 的 相关 信息 ， 如 图 2-42 所 示 。 


CE D7 6) wos error 
ZN FOE) EDV ci TR We 


划 当 强求 水 dHd 网 一 


© 


‘WampServer 
Voum 06 mag [cng Jossor Y) 


Server Configuration 
Apache Version: 
2423 ~ Documentation 


Php verson 
7040 - Documentation 


Server Sottware: 
Apache/2.4.23 (Wu2) PH/7.0.10 - Port defined for Apache: 80 


loaded Extensions 

















图 2-42 PHP 配置 环境 的 相关 信息 





2.6 ”案例 实战 一 一 测试 第 一 个 PHP 程序 


上 面 讲述 了 3 种 服务 器 环境 的 搭建 方法 ， 读 者 可 以 根据 自己 的 需求 进行 选择 安装 即 可 。 
建议 新 手 采 用 集成 开发 环境 。 

下 面 通过 一 个 实例 讲解 如 何 编写 PHP 程序 并 运行 查看 效果 。 下 面 以 WampServer 集成 开发 
环境 为 例 进 行 讲 解 。 读 者 可 以 使 用 任意 文本 编辑 软件 ， 如 记事 本 ， 新 建 名 称 为 “helloworld” 
的 文件 ， 如 图 2-43 所 示 。 输 入 以 下 代码 : 


<HTML> 

<HEAD> 

</HEAD> 

<BODY> 

<h2>PHP Hello World - 来 自 PHP 的 问候 。</h2> 

<?php 
echo "Hello, World.™"; 
echo "你 好 世界 。"; 

?> 

</BODY> 

</HTML> 
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将 文件 保存 在 C:vwampvwww 目录 下 ， 保 存 格 式 为 php。 在 浏览 器 的 地 址 栏 中 输入 
“http://localhost/helloworld.php”， 并 按 Enter 键 确认 ， 运 行 结 果 如 图 2-44 所 示 。 




















国 helcwodd.php -记事 本 - OO x © [OO nee D- ©) ccahon 四 “ 
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bP PHP Hello World - 米 自 PHP 的 问候 。 
他 Hello,Worle 你 好 世界 。 

《<h2>P 到 Hello World - 来 自 Pi 的 问候 。</h2> 

《Pphp 

echo “Hello, Yorld.” 

ocho“ 你 村 世界 。 

BODY 

| 

图 2-43 记事 本 窗口 图 2-44 ”程序 运行 效果 

【代码 剖析 】 


(1) “PHP Hello World - 来 自 PHP 的 问候 。” 是 HTML 中 的 <BODY><h2>PHP Hello 
World - 来 自 PHP 的 问候 。</h2></BODY> 所 生成 的 。 


(2) “Hello, World. 你 好 世界 。” 则 是 由 “<?php echo "Hello, World."; echo "你 好 世界 。 
"; ?>” 生 成 的 。 


(3) 在 HTML 中 嵌入 PHP 代码 的 方法 即 是 在 <?php ?> 标识 符 中 间 输 入 PHP 语句 ， 语 句 
要 以 “;” 号 结束 。 


(4) <?php ?> 标识 符 的 作用 就 是 告诉 Web 服务 器 ，PHP 代码 从 什么 地 方 开始 ， 到 什么 地 
方 结束 。<?php ?> 标识 符 内 的 所 有 文本 都 要 按照 PHP 语言 进行 解释 ， 以 区 别 于 HTML 代码 。 


2.7 疑难 解 惑 


疑问 1: 如 何 设置 网 站 的 主 目录 ? 


答 : 在 Windows 10 操作 系统 中 ， 设 置 网 站 主 目录 的 方法 如 下 。 

利用 本 章 的 方法 打开 “计算 机 管理 ”窗口 ， 选 择 Default Web Site 选项 ， 如 图 2-45 所 示 。 

在 右 侧 窗 格 中 单 击 “ 基 本 设置 ”文字 链接 ， 打 开 “ 编 辑 网 站 ”对 话 框 ， 单 击 “物理 路 
径 ” 文 本 框 右 侧 的 按钮 ， 即 可 在 打开 的 对 话 框 中 重新 设置 网 站 的 主 目录 ， 如 图 2-46 所 示 。 
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图 2-45 “计算 机 管理 ”窗口 图 2-46 “编辑 网 站 ”对 话 框 


疑问 2: 如 何 却 载 IIS? 


答 : 用 户 经 常会 遇 到 IIS 不 能 正常 使 用 的 情况 ， 所 以 需要 先 卸 载 TS， 然 后 再 次 安装 即 
可 。 利 用 本 章 的 方法 打开 “Windows 功能 ”对 话 框 ， 如 图 2-47 所 示 ， 取 消 选 中 Internet 
Information Services 选 框 ， 单 击 “确定 ”按钮 ， 系 统 将 自动 完成 IS 的 印 载 。 
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图 2-47 “Windows 功能 ”对 话 框 
疑问 3: 查看 PHP 文件 时 中 文 显 示 为 乱码 ， 而 英文 和 数字 显示 正常 ? 


答 : 在 使 用 下 浏览 器 查看 PHP 文件 时 ， 如 果 结 果 显 示 中 只 有 中 文 为 乱码 ， 而 英文 和 数 
字 显 示 正 常 ， 可 以 通过 修改 编码 方式 来 解决 问题 。 在 php.ini 配置 文件 中 找到 default_charset， 
将 其 值 设 置 为 gb2312 即 可 解决 问题 ， 如 图 2-48 所 示 。 


图 phpini - 记事 本 - 0O x 
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;http://php.net/default-mimetype ~ 
default_nimetype = “text/himl” 


; PHP's default character set is set to UIF-8. 
人 SoS default-charset 


1 PHP internal character encoding is set to empty. 
1 If empty, default_charset is used. 

;http://php. net/internal-encoding 
iinternal_encoding = 








; PHP input character encoding is set to empty. 
; If empty，default_charset is used. 
:http://php.net/input-encoding 
iinput_encoding = 


，PHP output character encoding is set to empty. 
1 If empty, default_charset is used. v 











2-48 ”php.ini 配置 文件 


第 3 重 
零 基础 开始 学 习 
一 一 PHP 的 
基本 语法 


上 一 章 讲述 了 PHP 环境 的 搭建 方法 ， 本 章 开始 学 习 PHP 的 基本 语法 ， 主 要 包 
括 PHP 的 标识 符 、 编 码 规范 、 常 量 、 变 量 、 数 据 类 型 、 运 算 符 和 表达 式 等 。 通 过 
本 章 的 学 习 ， 读 者 可 以 掌握 PHP 的 基本 语法 知识 和 技能 。 
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3.1 PHP 语言 标识 


作为 嵌入 式 脚 本 语言 ，PHP 是 以 <?php ?> 标识 符 为 开始 和 结束 标记 的 。 当 服务 器 解析 一 
个 PHP 文件 时 ， 通 过 寻找 开始 标记 和 结束 标记 ， 告 诉 PHP 开始 和 停止 解析 其 中 的 代码 。 凡 是 
标记 语言 以 外 的 内 容 都 会 被 PHP 解释 器 忽略 。 但 是 ，PHP 代码 有 不 同 的 标识 风格 。 下 面 学 习 
其 他 类 型 的 标识 风格 。 

1. 脚本 风格 

有 的 编辑 器 对 PHP 代码 完全 采用 另外 一 种 表示 方式 ， 比 如 <script></scripf> 的 表示 方式 。 

例如 : 

<script language="php"> 

echo "这 是 PHP 的 script 表示 方式 。"; 

</script> 

这 种 表示 方式 十 分 类 似 于 HTML 页 面 中 JavaScript 的 表示 方式 。 

2. 简短 风格 

有 时 候 ， 读 者 会 看 到 一 些 代 码 中 出 现 用 <? ?> 标识 符 表示 PHP 代码 的 情况 。 这 种 就 是 所 谓 
的 “ 短 风 格 ”(Short Style) 表 示 法 。 例 如 : 

<? echo "这 是 PHP 短 风格 的 表示 方式 。"?> 

这 种 表示 方式 在 正常 情况 下 并 不 推荐 。 并 且 在 php.ini 文件 中 ，short_open tags 设置 默认 
是 关闭 的 。 另 外 ， 以 后 提 到 的 一 些 功能 设置 会 与 这 种 表示 方法 相 冲 突 ， 比 如 与 XML 的 默认 标 
识 符 相 冲突 。 

3. ASP 风格 

受 ASP 的 影响 ， 为 了 照顾 ASP 使 用 者 对 PHP 的 使 用 ，PHP 提供 了 ASP 标识 风格 。 例 如 : 





<g% 
echo "这 是 PHP 的 Ase 的 表示 方式 。"; 
%> 


这 种 表示 方式 是 在 特殊 情况 下 使 用 的 ， 正 常情 况 下 并 不 推荐 使 用 。 


3.2 ”熟悉 编码 规范 


由 于 现在 的 Web 开发 往往 是 多 人 一 起 合作 完成 的 ， 所 以 使 用 相同 的 编码 规范 显得 非常 重 
要 。 特 别 是 新 的 开发 人 员 参 与 时 ， 往 往 需要 知道 前 面 开发 的 代码 中 变量 或 函数 的 作用 等 。 如 
果 使 用 统一 的 编码 规范 ， 这 就 容易 多 了 。 


3.2.1 什么 是 编码 规范 


编码 规范 规定 了 某 种 语言 的 一 系列 默认 编程 风格 ， 用 来 增强 这 种 语言 的 可 读 性 、 规 范 性 
和 可 维护 性 。 编 码 规范 主要 包括 语言 下 的 文件 组 织 、 缩 进 、 注 释 、 声 明 、 空 格 处 理 、 命 名 规 
则 等 。 

遵守 PHP 编码 规范 有 以 下 好 处 。 

(1) 编码 规范 是 团队 开发 中 对 每 个 成 员 的 基本 要 求 。 对 编码 规范 遵循 得 好 坏 是 一 个 程序 员 
成 熟 程度 的 表现 。 

(2) 能 够 提高 程序 的 可 读 性 ， 利 于 开发 人 员 互 相交 流 。 

(3) 良好 一 致 的 编程 风格 在 团队 开发 中 可 以 达到 事半功倍 的 效果 。 

(4) 有 助 于 程序 的 维护 ， 可 以 降低 软件 成 本 。 


3.2.2 ”PHP 的 一 些 编码 规范 


PHP 作为 高 级 语言 的 一 种 ， 特 别 强 调 编码 规范 。 具 体 表现 在 以 下 几 个 方面 。 

1. 表述 

比如 在 PHP 的 正常 表述 中 ， 每 一 条 PHP 语句 都 是 以 “;” 结 尾 ， 这 个 规范 就 告诉 PHP 要 
执行 此 语句 。 例 如 : 


<?php 
echo "PHP 以 分 号 表示 语句 的 结束 和 执行 。"; 


时 

2. 注释 

在 PHP 语言 中 ， 常 见 的 注释 包括 以 下 几 种 风格 。 
(1) C 语 言 风 格 。 例 如 : 

/* 这 是 c 语言 风格 的 注释 内 容 */ 

这 种 方法 还 可 以 多 行使 用 。 例 如 : 

/* 这 是 

C 语 言 风格 

的 注释 内 容 

*/ 

注意 : 注释 不 能 柑 套 ， 因 为 PHP 不 进行 块 注释 的 嵌 套 检查 ， 所 以 以 下 写法 是 错误 的 
echo "这 里 开始 嵌 套 注释 "; /* 嵌 套 注释 时 PHP 会 报错 */ 
*/ 

(2) C++ 风格 。 例 如 : 


// 这 是 c++ 风格 的 注释 内 容 行 一 
// 这 是 c++ 风格 的 注释 内 容 行 二 
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这 种 方法 只 能 一 句 注 释 占用 一 行 。 使 用 时 可 单独 一 行 ， 也 可 以 使 用 在 PHP 语句 之 后 的 同 
一 行 。 

(3) Shell 风格 。 例 如 : 

# 这 是 shell 风格 的 注释 内 容 


这 种 方法 只 能 一 句 注释 占用 一 行 。 使 用 时 可 单独 一 行 ， 也 可 以 使 用 在 PHP 语句 之 后 的 同 
一 行 。 
3. 空白 


PHP 对 空格 、 回 车 造成 的 新 行 、Tab 等 留 下 的 空白 的 处 理 也 遵循 编码 规范 。PHP 对 它们 
都 进行 忽略 。 这 跟 浏 览 器 对 HTML 语言 中 的 空白 的 处 理 是 一 样 的 。 

合理 地 运用 空白 符 ， 可 以 增强 代码 的 清晰 性 和 可 读 性 。 

(1) 下 列 情况 应 该 总 是 使 用 两 个 空白 行 。 

Q 两 个 类 的 声明 之 间 。 

@ 一 个 源 文件 的 两 个 代码 片段 之 间 。 

(2) 下 列 情况 应 该 总 是 使 用 一 个 空白 行 。 

中 两 个 函数 声明 之 前 。 

@ 函数 内 的 局 部 变量 和 函数 的 第 一 个 语句 之 间 。 

@ 块 注释 或 单行 注释 之 前 。 

图 一 个 函数 内 的 两 个 逻辑 代码 段 之 间 。 

(3) 合理 利用 空格 可 以 通过 代码 的 缩 进 提高 可 读 性 。 

中 空格 通常 使 用 于 关键 字 与 括号 之 间 ， 但 是 ， 函 数 名 称 与 左 括号 之 间 不 使 用 空格 分 开 。 

@ 函数 参数 列表 中 的 逗号 后 面 通常 会 插入 空格 。 

@ for 语句 的 表达 式 应 该 用 逗号 分 开 ， 后 面 添加 空格 。 


4. 指令 分 隔 符 
在 PHP 代码 中 ， 每 个 语句 后 需要 用 分 号 结束 命令 。 一 段 PHP 代码 中 的 结束 标记 隐 含 表示 
了 一 个 分 号 ， 所 以 在 PHP 代码 段 中 的 最 后 一 行 可 以 不 用 分 号 结束 。 例 如 : 


<?php 

echo "这 是 第 一 个 语句 "; // 每 个 语句 都 加 入 分 号 

echo "这 是 第 二 个 语句 "; 

echo "这 是 最 后 一 个 语句 "?> ”// 结束 标记 “2?>” 隐 含 了 分 号 ， 这 里 可 以 省 略 分 号 








5. 与 HTML 语言 混合 搭配 


凡是 在 一 对 PHP 开始 和 结束 标记 之 外 的 内 容 都 会 被 PHP 解析 器 忽略 ， 这 使 得 PHP 文件 
可 以 具备 混合 内 容 。 可 以 使 PHP 嵌入 到 HIML 文档 中 去 。 例 如 : 


<HTML> 

<HEAD> 

<TITLE>PHP 与 HTML 混合 搭配 </TITLE> 
</HEAD> 

<BODY> 


<?php 
echo "嵌入 的 PHP 代码 "; 
全 学 
</BODY> 
</HTML> 


33 党 量 
常量 和 变量 是 构成 PHP 程序 的 基础 。 下 面 讲述 如 何 声 明和 使 用 常量 。 


3.3.1 ”声明 和 使 用 常量 


在 PHP 中 ， 常 量 是 一 旦 声明 就 无 法 改变 的 值 。PHP 通过 define0 命 令 来 声明 常量 。 其 语 
法 格式 如 下 : 
define ("常量 名 "， 常 量 值 , mixed value); 


常量 名 是 一 个 字符 串 ， 往 往 在 PHP 编码 规范 的 指导 下 使 用 大 写 的 英文 字符 来 表示 。 例 如 
CLASS_NAME、MYAGE 等 。 常 量 值 也 可 为 表达 式 。mixed value 是 可 选 参数 ， 表 示 常 量 的 名 
字 是 否 区 分 大 小 写 ， 如 果 设 置 为 tue， 则 表示 不 区 分 大 小 写 ， 默 认为 false。 

常量 值 可 以 是 很 多 种 PHP 数据 类 型 的 ， 可 以 是 数组 ， 也 可 以 是 对 象 ， 当 然 还 可 以 是 字符 
和 数字 。 

常量 就 像 变 量 一 样 储 存 数 值 。 但 是 ， 与 变量 不 同 的 是 ， 常 量 的 值 只 能 设 定 一 次 ， 并 且 无 
论 在 代码 的 任何 位 置 ， 它 都 不 能 被 改动 。 

常量 声明 后 具有 全 局 性 ， 函 数 内 外 都 可 以 访问 。 

【 例 3.1】 声 明和 使 用 常量 (示例 文件 ch03\3.1.php)。 

<?ph] 

ER "日 出 江 花红 胜 火 ， 春 来 江水 绿 如 蓝 。") ; // 定 义 常 量 SHIGE 


echo SHIGE; // 输 出 常量 SHIGE 
2> 


程序 运行 结果 如 图 3-1 所 示 。 一 去 
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【案例 剖析 】 机 则 。 二 BV) 可 王 e/A) 工具 (部 且 人) 
日 出 江 花 红 胜 火 ， 春 来 江水 绿 如 蓝 。 
(1) 用 define 函数 声明 一 个 常量 SHIGE。 而 常量 的 全 局 性 
体现 在 可 在 函数 内 外 进行 访问 。 下 让: 





(2) 常量 只 能 储存 布尔 值 、 整 型 、 浮 点 型 和 字符 串 数据 。 


3.3.2 ”使 用 系统 预定 义 常量 


PHP 的 系统 预定 义 常量 是 指 PHP 在 语言 内 部 预先 定义 好 的 一 些 常量 。PHP 中 预定 了 很 多 
系统 内 置 常量 ， 这 些 常 量 可 以 被 随时 调用 。 例 如 ， 下 面 是 一 些 常见 的 内 置 常量 。 


图 3-1 声明 和 使 用 常量 
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1._FILE_ 


这 个 默认 常量 是 PHP 程序 文件 名 。 若 引用 文件 (include 或 require)， 则 在 引用 文件 内 的 该 
常量 为 引用 文件 名 ， 而 不 是 引用 它 的 文件 名 。 


2.LINE_ 


这 个 默认 常量 是 PHP 程序 行 数 。 若 引用 文件 (include 或 require)， 则 在 引用 文件 内 的 该 常 
量 为 引用 文件 的 行 数 ， 而 不 是 引用 它 的 文件 的 行 数 。 


3. PHP_VERSION 
这 个 内 置 常量 是 PHP 程序 的 版 本 ， 如 3.0.8-dev。 

4.PHP_OS 

这 个 内 置 常 量 指 执行 PHP 解析 器 的 操作 系统 名 称 ， 如 Linux。 
5. TRUE 

这 个 常量 就 是 真 值 (true)。 

6. FALSE 

这 个 常量 就 是 伪 值 (false)。 

7.E_ERROR 

这 个 常量 指 到 最 近 的 错误 处 。 

8.E_ WARNING 

这 个 常量 指 到 最 近 的 警告 处 。 

9.E_PARSE 

该 常量 为 解析 语法 有 潜在 问题 处 。 

10.E NOTICE 


这 个 量 为 发 生 异 常 (但 不 一 定 是 错误 ) 处 。 例 如 存 取 一 个 不 存在 的 变量 。 
下 面 举例 说 明 系 统 常量 的 使 用 方法 。 
【 例 3.2】 使 用 系统 内 置 常量 (示例 文件 ch03\3.2.php)。 





<?php 

echo (_FILE ); // 输出 文件 的 路 径 和 文件 名 
echo "<br/>"; // 输出 换行 
echo(_LINE ); // 输出 语句 所 在 的 行 数 


echo "<br/>"; 

echo (PHP_VERSION); // 输出 PHP 的 版 本 
echo "<br/>"; 

echo (PHP_OS); // 输出 操作 系统 名 称 


2 











程序 运行 结果 如 图 3-2 所 示 。 © [SE TT Br | 
【案例 剖析 】 aad 
(1) echo "<p>" 语 句 表示 为 输出 换行 。 i 
(2) ”echo(_FILE ) 语 句 输出 文件 的 文件 名 ， 包 括 详 wv 
细 的 文件 路 径 。echo(_LINE ) 语 句 输出 该 语句 所 在 的 行 Eo 





数 。echo(PHP_VERSION) 语 句 输出 PHP 程序 的 版 本 。 


echo(PHP_OS) 语 句 输出 执行 PHP 解析 器 的 操作 系统 名 称 。 3-2 使 用 内 置 常量 


3.4 变 量 


变量 像 是 一 个 贴 有 名 字 标 签 的 空 盒子 。 不 同 的 变量 类 型 对 应 不 同 种 类 的 数据 ， 就 像 不 同 
种 类 的 东西 要 放 入 不 同 种 类 的 盒子 一 样 。 


3.4.1 PHP 中 的 变量 声明 


与 C 语言 或 Java 语言 不 同 的 是 ，PHP 中 的 变量 是 弱 类 型 的 。 在 C 语言 或 Java 语言 中 ， 
需要 对 每 一 个 变量 声明 类 型 ， 而 在 PHP 中 不 需要 这 样 做 。 这 是 极其 方便 的 。 

PHP 中 的 变量 一 般 以 $ 作 为 前 级 ， 然 后 以 字母 a 一 z 的 大 小 写 或 者 下 划 线 “_ ”开头 。 这 是 
变量 的 一 般 表 示 。 

合法 的 变量 名 可 以 是 : 

$hello 


$Aforml 
$_formhandler 


非法 的 变量 名 如 : 

$168 

$11ike 

PHP 中 不 需要 显 式 地 声明 变量 ， 但 是 定义 变量 前 进行 声明 并 带 有 注释 ， 这 是 一 个 好 的 程 
序 员 应 该 养 成 的 习惯 。PHP 的 赋值 有 两 种 ， 包 括 传 值 和 引用 ， 它 们 的 区 别 如 下 。 

(1) 传 值 赋值 ， 使 用 “=” 直 接 将 赋值 表达 式 的 值 赋 给 另 一 个 变量 。 

(2) 引用 赋值 ， 将 赋值 表达 式 内 存 的 空间 的 引用 赋 给 另 一 个 变量 。 需 要 在 “=” 左 右 的 变 
量 前 面 加 上 一 个 “&” 符 号 。 在 使 用 引用 赋值 的 时 候 ， 两 个 变量 将 会 指向 内 存 中 同一 个 存储 
空间 ， 所 以 任意 一 个 变量 的 变化 都 会 引起 另外 一 个 变量 的 变化 。 

【 例 3.3】 使 用 传 值 赋值 和 引用 赋值 (示例 文件 ch03\3.3.php)。 


<?php 


echo "使 用 传 值 方式 赋值 : <br/>"; // 输出 “使 用 传 值 方式 赋值 : ” 

$a = " 稻 云 不 雨 不 多 黄 "; 

$b = $a; // 将 变量 $a 的 值 赋值 给 sb， 两 个 变量 指向 不 同 内存 空 间 
echo "变量 a 的 值 为 " .$a."<br/>"; // 输出 变量 a 的 值 
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echo "变量 b 的 值 为 " .$b."<br/>"; // 输出 变量 b 的 值 

$a = "荞麦 空 花 早 着 霜 "; // 改变 变量 a 的 值 ， 变 量 b 的 值 不 受 影响 

echo "变量 a 的 值 为 " .$a."<br/>"; // 输出 变量 a 的 值 

echo "变量 b 的 值 为 ". $b."<p>"; // 输 出 变量 b 的 值 

echo "使 用 引用 方式 赋值 : <br/>"; // 输 出 ”使 用 引用 方式 赋值 

$a = "已 分 忍 饥 度 残 岁 "7 

$b = &$a; // 将 变量 $a 的 引用 赋 给 $b， 两 个 变量 指向 同一 块 内 存 空间 

echo "变量 a 的 值 为 " . $a."<br/>"; // 输出 变量 a 的 值 

echo "变量 b 的 值 为" . $b."<br/>"; // 输出 变量 b 的 值 

$a = "更 堪 岁 里 头 添 长 "; 

A 

改变 变量 a 在 内 存 空间 中 存储 的 内 容 ， 变 量 b 也 指向 该 空间 ，b 的 值 也 发 生变 化 

ad 

echo "变量 a 的 值 为 ". $a."<br/>"; // 输出 变量 a 的 值 

echo "变量 b 的 值 为 ". $b." <br/>"; // 输出 变量 b 的 值 

总 

程序 运行 结果 如 图 3-3 所 示 。 一 到 
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342 可 变 变 量 和 变量 的 引用 2 


一 般 的 变量 很 容易 理解 ， 但 是 有 两 种 变量 的 概念 比较 
难以 理解 ， 这 就 是 可 变 变量 和 变量 的 引用 。 下 面 通过 示例 
对 它们 进行 介绍 。 

【 例 3.4】 使 用 可 变 变量 和 变量 的 引用 (示例 文件 
ch03\3.4.php)。 





<?php 

$value0 = "guest"; // 定 义 变量 $value0 并 赋值 
$$value0 = "customer"; // 再 次 给 变量 赋值 

echo $guest."<br/>"; // 输出 变量 


$guest = " 张 飞 "; // 定义 变量 $guest 并 赋值 
echo $guest."\t".$$value0."<br/>"; 

$valuel = "王小明 "; // 定义 变量 $valuel 
$value2 = &$valuel; // 引用 变量 并 传递 变量 
echo $valuel."\t".$value2."<br/>"; 

$value2 = " 李 丽 "; 

echo $valuel."\t".$value2; 

人 


程序 运行 结果 如 图 3-4 所 示 。 
【案例 剖析 】 


(1) 在 代码 的 第 一 部 分 中 ，S$value0 被 赋值 为 “guest” 





使 用 传 值 方式 赋值 ^ 


的 值 为 养 麦 空 花 | 
变量 b 的 信 为 和 不 雨 不 多 黄 


使 用 引用 方式 赋值 : 

变量 a 的 值 为 已 分 妨 饥 度 残 岁 
变量 b 的 值 为 已 分 起 饥 度 残 岁 
变量 a 的 值 为 更 堪 岁 里 问 添 长 
变量 b 的 值 为 更 堪 岁 里 问 添 长 
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图 3-3 程序 运行 结果 








EEC 
| € Ka) [Cn or Te x 
A WED EE ws IAT Mm 

i 

张 飞 张 

王 小 中 小 明 

李 丽 李 丽 


全 1 





3-4 ”使 用 可 变 变量 和 变量 的 引用 


。 而 $value0 相当 于 guest， 则 


$$value0 相当 于 $guest。 所 以 当 $$value0 被 赋值 为 “customer” 时 ， 打 印 $guest 就 得 到 customer。 
反之 ， 当 $guest 变量 被 赋值 为 “ 张 飞 ” 时 ， 打 印 $$value0 同样 得 到 “ 张 飞 ”。 这 就 是 可 变 变 量 。 
(2) 在 代码 的 第 二 部 分 中 ，$valuel 被 赋值 为 “王小明 ”， 然 后 通过 & 引 用 变量 $valuel 并 
赋值 给 $value2。 而 这 一 步 的 实质 是 ， 给 变量 $valuel 添加 了 一 个 别名 $value2。 所 以 打印 时 ， 都 
得 出 原始 赋值 “王小明 ”。 由 于 $value2 是 别名 ， 与 $valuel 指 的 是 同一 个 变量 ， 所 以 $value2 


被 赋值 为 “ 李 丽 ” 后 ，$valuel 和 $value2 都 得 到 新 值 “ 李 丽 ”。 

(3) 可 变 变 量 其 实 是 允许 改变 一 个 变量 的 变量 名 。 人 允许 使 用 一 个 变量 的 值 作为 另外 一 个 
变量 的 名 。 

(4) 变量 引用 相当 于 给 变量 添加 了 一 个 别名 。 用 & 来 引用 变量 ， 其 实 两 个 变量 名 指 的 都 
是 同一 个 变量 。 就 像 是 给 同一 个 盒子 贴 了 两 个 名 字 标 签 ， 两 个 名 字 标 签 指 的 都 是 同一 个 盒子 。 


3.4.3 变量 作用 域 


所 谓 变量 作用 域 (Scope)， 是 指 特定 变量 在 代码 中 可 以 被 访问 到 的 位 置 。 在 PHP 中 ， 有 6 
种 基本 的 变量 作用 域 法 则 。 

(1) 内 置 超 全 局 变量 : 在 代码 中 的 任意 位 置 都 可 以 访问 到 。 

(2) 常数 : 一 旦 声明 ， 它 就 是 全 局 性 的 。 可 以 在 函数 内 外 使 用 。 

(3) 全 局 变量 : 在 代码 中 声明 ， 可 在 代码 中 访问 ， 但 是 不 能 在 函数 内 访问 。 

(4) 在 函数 中 声明 为 全 局 变量 的 变量 : 就 是 同名 的 全 局 变量 。 

(5) 在 函数 中 创建 和 声明 为 静态 变量 的 变量 ; 在 函数 外 是 无 法 访问 的 ， 但 是 这 个 静态 变量 
的 值 是 可 以 保留 的 。 

(6) 在 函数 中 创建 和 声明 的 局 部 变量 : 在 函数 外 是 无 法 访问 的 ， 并 且 在 本 函数 终止 时 终止 
并 退出 。 

1. 超 全 局 变量 


超 全 局 变量 的 英文 是 Superglobal 或 者 Autoglobal( 自 动 全 局 变量 )。 这 种 变量 的 特性 是 : 不 
管 在 程序 的 任何 地 方 ， 也 不 管 是 函数 内 或 是 函数 外 ， 都 可 以 访问 。 而 这 些 “ 超 全 局 变量 ”就 
是 由 了 PHP 预先 定义 好 以 方便 使 用 的 。 

这 些 超 全 局 变量 或 自动 全 局 变量 介绍 如 下 。 
$GLOBALS: 包含 全 局 变量 的 数组 。 
$_GET: 包含 所 有 通过 GET 方法 传递 给 代码 的 变量 的 数组 。 
$_POST: 包含 所 有 通过 POST 方法 传递 给 代码 的 变量 的 数组 。 
$_FILES: 包含 文件 上 传 变量 的 数组 。 
$_COOKIE: 包含 cookie 变量 的 数组 。 
$_SERVER: 包含 服务 器 环境 变量 的 数组 。 
$_ENV: 包含 环境 变量 的 数组 。 
$_REQUEST: 包含 用 户 所 有 输入 内 容 的 数组 (包括 $ GET、$_POST 和 $_COOKIE)。 
$_SESSION: 包含 会 话 变 量 的 数组 。 


2. 全 局 变量 

全 局 变量 其 实 就 是 在 函数 外 声明 的 变量 ， 在 代码 中 都 可 以 访问 。 但 是 在 函数 内 是 不 能 访 
问 的 。 这 是 因为 函数 默认 就 不 能 访问 在 其 外 部 的 全 局 变量 。 

下 面 通过 例子 介绍 全 局 变量 的 使 用 方法 和 技巧 。 

【 例 3.5】 使 用 全 局 变量 (示例 文件 ch03\3.5.php)。 


讲 斑 诗 须 否 dHd 一 冯 站 淮 书 宇 朵 遇 册 人 洲 全 


N - 





PHP+MySQL 动态 网 站 开发 
案例 课堂 (第 2 版 ) B… 


<?php 
$price = 3; // 定 义 全 局 变量 
function showprice(){ 

echo $price; // 函 数 内 调用 全 局 变量 
showprice () 7 // 运 行 函数 
echo “苹果 的 价格 为 '.$price- ' 元 一 公斤 7 


Fe 


程序 运行 结果 如 图 3-5 所 示 。 | = 了 | 
【案例 剖析 】 加 日 ”站 在， 榴 宣 R(A) 工具 (1) 大 了 CH) - 
出 现 上 述 结 果 ， 是 因为 函数 无 法 访问 到 外 部 全 局 变 | [一 
量 ， 但 是 在 代码 中 都 可 以 访问 全 局 变量 。 | a 
如 果 想 让 函数 访问 某 个 全 局 变量 ， 可 以 在 函数 中 | ni nem em 
通过 global 关键 字 来 声明 。 训 是 说 ， 要 告诉 了 数 ， 它 。 ne 。 
要 调用 的 变量 是 一 个 已 经 存在 或 者 即将 创建 的 同名 全 
局 变量 ， 而 不 是 默认 的 本 地 变量 。 图 3-5 使 用 全 局 变量 
下 面 通过 例子 介绍 使 用 global 关键 字 的 方法 和 
技巧 。 
【 例 3.6】 使 用 global 关键 字 ( 示 例文 件 ch03\3.6.php)。 
<?php 
$price = 3; // 定 义 全 局 变量 


function showprice(){ 
global price; // 函 数 内 声明 全 局 变量 
} 
showprice(); 
echo "苹果 的 价格 为 '$price. ' 元 一 公斤 '; 


> 
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程序 运行 结果 如 图 3-6 所 示 。 
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文件 (有 六 各 6 查看) 收藏 交 (A) 工具 (TD 帮助 (H) 


苹果 的 价格 为 3 元 一 公斤 

















瑟 125% > 
图 3-6 使 用 global 关键 字 
另外 ， 读 者 还 可 以 通过 超 全 局 变量 中 的 SGLOBALS 数组 进行 访问 。 


下 面 通 过 例子 介绍 如 何 使 用 GLOBALS 数组 。 
【 例 3.7】 使 用 SGLOBALS 数组 (示例 文件 ch03\3.7.php)。 





<?php 
$aa = ' 九 曲 黄河 万 里 沙 '; // 定义 全 局 变量 
function showss(){ 
Saa — SGLOBALS [aa ls // 通过 $GLOBALS 数组 访问 全 局 变量 


echo $aa.'<br/>'; 


后 ， 


3 
showss () 7 
echo $aa.', 浪 淘 风 煞 自 天 涯 '; 


程序 运行 结果 如 图 3-7 所 示 。 
3. 静态 变量 


静态 变量 只 在 函数 内 存在 ， 函 数 外 无 法 访问 。 但 是 执行 
其 值 保 留 。 也 就 是 说 ， 这 一 次 执行 完毕 后 ， 这 个 静态 变 


量 的 值 保留 ， 下 一 次 再 执行 此 函数 ， 这 个 值 还 可 以 调用 。 


S$aa, 


下 面 通过 例子 介绍 静态 变量 的 使 用 方法 和 技巧 。 
【 例 3.8】 使 用 静态 变量 (示例 文件 ch03\3.8.php)。 


<?php 

$aa = 15; 

function sumaa(){ 

static $aa = 6;  // 初 始 化 静态 变量 
$aat+? 

echo ' 变 量 aa 的 值 为 : ' .$aa.'<br/>'; 
于 

Sumaa() 7 

echo $aa.'<br/>'; 

sumaa (); 

人 


程序 运行 结果 如 图 3-8 所 示 。 
【案例 剖析 】 


(1) 其 中 函数 外 的 echo 语句 无 法 调用 函数 内 的 static 
它 调 用 的 是 $aa = 15。 
(2) 另外 ，sumaa() 函 数 被 执行 了 两 次 ， 这 个 过 程 中 ， 


static $aa 的 运算 值得 以 保留 ， 并 且 通 过 $aa++ 进 行 了 累加 。 
3.4.4 变量 的 销毁 

当 用 户 创建 一 个 变量 时 ， 相 应 地 在 内 存 中 有 一 个 空间 专门 用 于 存储 该 变量 ， 该 空间 引用 
计数 加 1。 当 变量 与 该 空间 的 联系 被 断 开 时 ， 则 空间 引用 计数 减 1， 直 到 引用 计数 为 0， 则 成 
为 垃圾 。 


PHP 有 自动 回收 垃圾 的 机 制 ， 用 户 也 可 以 手动 销毁 变量 ， 通 常 使 用 unsetO 函 数 来 实现 。 
该 函数 的 语法 格式 如 下 : 


void unset (变量 ) 


= 
[@O- Oreecaho-. P - © [Bocalhor 
Er 
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图 3-7 使 用 GLOBALS 数组 
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变量 aa 的 值 为 : 8 
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图 3-8 使 用 静态 变量 





其 中 ， 如 果 变 量 类 型 为 局 部 变量 ， 则 变量 被 销毁 ;如 果 变量 类 型 为 全 局 变量 ， 则 变量 不 会 被 
销毁 。 


【 例 3.9】 使 用 unsetO( 示 例文 件 ch03\3.9.php)。 
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<?php 
function showshi(){ 
// 声 明 函 数 
global $aa7 
// 函 数 内 使 用 global 关键 字 声 明 全 局 变量 $aa 
unset ($aa); 二 二 口 x 
// 使 用 unset () 销毁 不 再 使 用 的 变量 $aa © EB po Bien 
六 HF 入 后 丰 ”可 看 VM 收 茂 夫 AI 工 县 加 划 [H) 
$aa= 如今 直 上 银河 去 ， 同 到 牵 牛 织女 家 。 "7 人 
// 函 数 外 声明 全 局 变量 
showshi (); // 调 用 函数 
echo $aa; // 查 看 全 局 变量 是 否 发 生变 化 


125% 7 
流光 


程序 运行 结果 如 图 3-9 所 示 。 


3.5 理解 变量 的 类 型 


从 PHP 4 开始 ，PHP 中 的 变量 不 需要 事先 声明 ， 赋 值 即 声明 。 声 明和 使 用 这 些 数据 类 型 
前 ， 读 者 需要 了 解 它们 的 含义 和 特性 。 


3.5.1 什么 是 类 型 


不 同 的 数据 类 型 其 实 就 是 所 储存 数据 的 不 同 种 类 。PHP 主要 有 下 列 数据 类 型 。 

(1) 整 型 (integeD: 用 来 储存 整数 。 

(2) 浮 点 型 (float 或 double): 用 来 储存 实数 。 

(3) 字符 串 型 (string): 用 来 储存 字符 串 。 

(4) 布尔 型 (boolean): 用 来 储存 真 (true) 或 假 (false)。 

(5) 数组 型 (array): 用 来 储存 一 组 数据 。 

(6) 对 象 型 (object): 用 来 储存 一 个 类 的 实例 。 

作为 弱 类 型 语言 ，PHP 也 被 称 为 动态 类 型 语言 。 在 强 类 型 语言 (如 C 语言 ) 中 ， 一 个 变量 
只 能 储存 一 种 类 型 的 数据 ， 并 且 这 个 变量 在 使 用 前 必须 声明 变量 类 型 。 而 在 PHP 中 ， 给 变量 
赋 什 么 类 型 的 值 ， 这 个 变量 就 是 什么 类 型 的 。 例 如 以 下 几 个 变量 : 

$hello = 'hello world'; 

// 由 于 'hello world' 是 字符 串 ， 则 变量 Shello 的 数据 类 型 就 为 字符 串 类 型 

hello = 100; 

于 100 为 整 型 ， 所 以 $hello 也 就 为 整 型 


$wholeprice = 100.0; 
// 由 于 100.0 为 浮 点 型 ， 所 以 $wholeprice 就 是 浮 点 型 


由 此 可 见 ， 对 变量 而 言 ， 如 果 没 有 定义 变量 的 类 型 ， 则 它 的 类 型 由 所 赋值 的 类 型 来 决定 。 


3.5.2 ” 整 型 


整 型 是 数据 类 型 中 最 为 基本 的 类 型 。 在 32 位 的 运算 中 ， 整 型 的 取 值 范围 是 -2147483648 
~+2147483647。 整 型 可 以 表示 为 十 进 制 、 十 六 进 制 和 八进制 数 形式 。 


3-9 使 用 unset() 





例如 : 
3560 // 十 进 制 整数 
01223 // 八 进 制 整数 


0x1223 // 十 六 进 制 整数 


35.3 浮 点 型 


浮 点 型 就 是 实数 。 在 大 多 数 运行 平台 下 ， 这 个 数据 类 型 的 大 小 为 8 个 字 节 。 它 的 近似 取 
值 范围 是 2.2E-308 ~ 1.8E+308( 科 学 计数 法 )。 
例如 : 


-1.432 
1E+07 
0.0 


3.5.4 布尔 型 


布尔 型 只 有 两 个 值 ， 就 是 true 和 false。 布 尔 型 是 十 分 有 用 的 数据 类 型 ， 通 过 它 ， 程 序 实 
现 了 逻辑 判断 的 功能 。 

而 对 于 其 他 数据 类 型 ， 基 本 都 有 布尔 属性 。 

(1) 整 型 : 为 0 时 ， 其 布尔 属性 为 false， 为 非 零 值 时 ， 其 布尔 属性 为 true。 

(2) 浮 点 型 : 为 0.0 时 ， 其 布尔 属性 为 false， 为 非 零 值 时 ， 其 布尔 属性 为 true。 

(3) 字符 串 型 : 空 字 符 串 “”， 或 者 零 字 符 串 “0” 时 ， 为 false， 包 含 除 此 以 外 的 字符 串 
时 为 true。 

(4) 数组 型 : 若 不 含 任何 元 素 ， 为 false， 只 要 包含 元 素 ， 则 为 true。 

(5) 对 象 型 、 资 源 型 : 永远 为 true。 

(6) 空 型 : 永远 为 false。 


3.5.5 “字符 串 型 


字符 串 型 的 数据 是 引号 之 间 的 一 串 字 符 。 引 号 有 双 引 号 "和 单 引 号 "两 种 。 

但 是 这 两 种 表示 也 有 一 定 的 区 别 。 

双 引 号 几乎 可 以 包含 所 有 的 字符 。 但 是 其 中 显示 的 是 变量 的 值 ， 而 不 是 变量 的 名 。 而 有 
些 特殊 字符 在 使 用 时 需要 加 上 “\” 这 一 转 义 符号 。 

单 引号 内 的 字符 是 被 直接 表示 出 来 的 。 

下 面 通过 一 个 例子 来 讲述 上 面 几 种 类 型 的 使 用 方法 和 技巧 。 

【 例 3.10】 使 用 各 种 数据 类 型 (示例 文件 ch03\3.10.php)。 


<?php 

Sintl = 2018; 

$int2 = 01223; ”// 八 进 制 整数 
$int3 = 0x1223; ”// 十 六 进 制 整 数 
echo "输出 整数 类 型 的 值 ，"; 
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echo $intl; 

eeho “NE // 输 出 一 个 制 表 符 
echo $int2; // 输 出 659 

echo "\t"; 

echo $int3;  // 输 出 4643 

echo “<br>"» 

$floatl = 54.66; 

echo $floatl; // 输 出 54.66 
echo "<br>"; 

echo "输出 布尔 型 变量 : "7 

echo (Boolean) ($int1); // 将 intl 整 型 转化 为 布尔 变量 
echo "<br> 

$stringl = "字符 串 类 型 的 变量 "; 
echo $stringl; 

2 








程序 运行 结果 如 图 3-10 所 示 。 | E 
[@@- rp//ocaho. ~ | @ locahont 
| zsm 鲍 中 日 豆 看 (V) 收 划 夫 A) 工具 Mm 帮 卫 (H) 








3.5.6 ”数组 型 移出 攻关 型 的 作 ，2018 659 4 和 
输出 布尔 型 变量 : 
数组 是 PHP 变量 的 集合 ， 它 是 按照 “ 键 ”与 “ 值 ” | 从 员 天 开 的 要 是 | 





对 应 的 关系 组 织 数据 的 。 数 组 的 键 可 以 是 整数 ， 也 可 以 

















是 字符 串 。 R125% 7 
在 默认 情况 下 ， 数 组 元 素 的 键 为 从 0 开始 的 整数 。 图 3-10 使 用 各 种 数据 类 型 
在 PHP 中 ， 使 用 list0 函 数 或 aray0 函 数 来 创建 数 
组 ， 也 可 以 直接 进行 赋值 。 
下 面 使 用 arrayO 函 数 创建 数组 。 
【 例 3.11】 使 用 array0 函 数 创建 数组 (示例 文件 ch03\3.11.php)。 
<?php 
$arr=array // 定义 数组 并 赋值 
( 
0=>" 黑 梅 "， 
2=>" 元 代 : 王 冕 "， 
1->" 香 家 洗 砚 池 头 树 ， 个 个 花 开 淡 墨 痕 。" 
3=>" 不 要 人 寺 好 颜色 ， 只 留 清 气 满 乾坤 。" 
jw 
for ($i=0; $i<count (Sarr); $i++)  // 使 用 for 循环 输出 数组 的 内 容 
和. 
$arrl = each ($arr); 
echo "$arrl[value] <br>"; 
程序 运行 结果 如 图 3-11 所 示 。 @ [Eeevhecahe. P - OT pr 
【案例 剖析 】 a seal) BEV AIR MW 
(1) 程序 中 用 => 为 数组 赋值 ， 数 组 的 下 标 只 是 存储 的 人 人 全 臣下 和 人 
标识 ， 没 有 任何 其 他 意义 。 数 组 元 素 的 排列 以 加 入 的 先后 。 “全 
顺序 为 准 。 二 
本 图 3-11 使 用 array() 函 数 创建 数组 


G4 


返 


可 
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(2) 本 程序 采用 for 循环 语句 输出 整个 数组 ， 其 中 count 函数 返回 数组 的 个 数 ，each 函数 
回 当前 数组 指针 的 索引 / 值 对 ， 后 面 章节 中 将 详细 讲述 函数 的 使 用 方法 。 





例 3.11 中 的 语句 可 以 简化 如 下 。 

【 例 3.12】 使 用 array0 函 数 的 简化 语句 (示例 文件 ch03\3.12.php)。 

<?php 

$arr = array (" 墨 梅 ", "元 代 : 王 晃 "," 吾 家 洗 砚 池 头 树 ， 个 个 花 开 淡 墨 痕 。"， "不 要 人 夸 好 颜色 ， 
只 留 清 气 满 乾坤 。") ; 

for ($i=0; $i<4; $i++) 

{ 

echo $arrlsil"<br>" 
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二 


程序 运行 结果 如 图 3-12 所 示 。 
从 结果 可 以 看 出 ， 两 种 写法 的 运行 结果 完全 一 样 。 
另外 ， 读 者 还 可 以 对 数组 的 元 素 一 个 一 个 地 赋值 ， 下 面 举例 说 明 。 上 面 的 程序 中 的 语句 


以 写成 如 下 形式 。 
【 例 3.13】 通 过 逐个 赋值 的 方式 创建 数组 (示例 文件 ch03\3.13.php)。 
<?php 
$arr[0] =" 墨 梅 "; // 对 数组 元 素 分 别 赋值 
$arr[2] ”元 代 : 王 黑 "; 


$arr [1] = " 香 家 洗 砚 池 头 树 ， 个 个 花 开 淡 黑 痕 。"j 

$arr[3] = "不 要 人 夸 好 颜色 ， 只 留 清 气 满 乾 坤 。"; 

for ($i=0; $i<count ($arr); $i++)  // 使 用 for 循环 输出 数组 的 内 容 
$arrl = each ($arr); 

echo "$arrl[value]<br>"; 


k 人 2 


程序 运行 结果 如 图 3-13 所 示 。 

















-EE a 
-roocato.. P» ©||B localhost x @ EO p/ho. P- ©| Blocalhost -时 
立 引 朋 纺 溉 ( 昌 理 看 V) 此 大 工具 二 (H) Eo 工 RD 基 有 0 
黑 梅 墨 梅 
元 代 : 王 晶 元 代 : 王 蝎 
吾 家 洗 砚 池 头 树 ， 个 个 花 开 淡 墨 痕 。 在 家 洗 砚 池 头 树 ， 个 个 花 开 淡 黑 痕 。 
不 要 人 寺 好 颜色 ， 只 留 清 气 满 阔 坤 。 不 要 人 夸 好 颜色 ， 只 留 清 气 满 乾坤 。 
导 125% ~ 加 125% > 
图 3-12 简化 后 的 程序 运行 结果 图 3-13 ”逐个 赋值 时 的 程序 运行 结果 
从 结果 可 以 看 出 ， 一 个 一 个 赋值 的 方法 与 上 面 两 种 写法 的 运行 结果 是 一 样 的 。 
5.7 对象 型 


对 象 就 是 类 的 实例 。 当 一 个 类 被 实例 化 以 后 ， 这 个 生成 的 对 象 被 传递 给 一 个 变量 ， 这 个 


变量 就 是 对 象 型 变量 。 对 象 型 变量 也 属于 资源 型 变量 。 


济 销 诗 采 下 dHd 一 一 了 当 幅 浸 妆 新 膝 谢 册 8 小 音 
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3.5.8 NULL 型 


NULL 类 型 用 来 标记 一 个 变量 为 空 。 但 一 个 空 字符 串 与 一 个 NULL 是 不 同 的 。 在 数据 库 
存储 时 ， 会 把 空 字符 串 和 NULL 区 分 开 处 理 。NULL 型 在 布尔 判断 时 永远 为 false。 很 多 情况 
下 ， 在 声明 一 个 变量 的 时 候 可 以 直接 先 赋值 为 NULL 型 ， 如 $value = NULL。 


3.5.9 资源 类 型 


资源 类 型 (Resources) 是 十 分 特殊 的 数据 类 型 ， 表 示 了 PHP 的 扩展 资源 。 它 可 以 是 一 个 打 
开 的 文件 ， 也 可 以 是 一 个 数据 库 连接 ， 甚 至 还 可 以 是 其 他 数据 类 型 。 但 是 在 编程 过 程 中 ， 资 
源 类 型 却 是 很 难 接触 到 的 。 


3.5.10 ”数据 类 型 之 间 的 相互 转换 

数据 从 一 个 类 型 转换 到 另外 一 个 类 型 ， 就 是 数据 类 型 转换 。 在 PHP 语言 中 ， 有 两 种 常见 
的 转换 方式 ， 自 动 数据 类 型 转换 和 强制 数据 类 型 转换 。 

1. 自动 数据 类 型 转换 


这 种 转换 方法 最 为 常用 。 直 接 输 入 数据 的 转换 类 型 即 可 。 

例如 ，float 型 转换 为 整数 int 型 ， 小 数 点 后 面 的 数 将 被 舍弃 。 如 果 float 数 超过 了 整数 的 
取 值 范围 ， 则 结果 可 能 是 0 或 者 整数 的 最 小 负数 。 

【 例 3.14】 自 动 数据 类 型 转换 (示例 文件 ch03\3.14.php)。 


<?php 

$floaa=8.88; // 定义 float 类 型 
echo (int)$floaa."<br/>"; // 转换 为 整数 类 型 输出 
$f1lobb=4E32; // 超过 整数 取 值 范围 


echo (int) $flobb; 
2 


程序 运行 结果 如 图 3-14 所 示 。 


| 过 口 x 
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图 3-14 ”自动 数据 类 型 转换 





2. 强制 数据 类 型 转换 
在 PHP 中 ， 可 以 使 用 setType0 函 数 强制 转换 数据 类 型 。 基 本 语法 格式 如 下 : 


Bool setType (var, string type) 


type 的 可 能 值 不 能 包含 资源 类 型 数据 。 


mi 


【 例 3.1S】 强 制 数据 类 型 转换 (示例 文件 ch03\3.15.php)。 
<?php EE | 
$floaa = 6.66; <¢ [IEE ry Ep 
echo setType ($floaa, "int")"<br/>"; 文件 ”后 二 喜 看 MW) 收 训 夫 工具 (帮助 由 
echo $floaa; 
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i 
程序 运行 结果 如 图 3-15 所 示 。 转 型 成 功 ， 则 返回 1， | 
否则 返回 0。 变 量 floaa 转换 为 整 型 后 为 6。 3-15 ”强制 数据 类 型 转换 


3.6 ”PHP7 的 新 变化 一 一 声明 标量 类 型 和 
函数 返回 值 类 型 
在 默认 情况 下 ， 所 有 的 PHP 文件 都 处 于 弱 类 型 校 验 模 式 。PHP 7 增加 了 标量 类 型 声明 的 


特性 。 标 量 类 型 声明 有 两 种 模式 ， 强制 模式 (默认 ) 和 严格 模式 。 

标量 类 型 声明 的 语法 格式 如 下 : 

declare (strict types=1); 

可 以 设置 strict_types 的 值 (1 或 者 0)，1 表示 严格 类 型 校 验 模式 ， 作 用 于 函数 调用 和 返回 
语句 ，0 表示 弱 类 型 校 验 模式 。 


Em 可 以 声明 标量 类 型 的 参数 类 型 包括 int、float、bool、string、interfaces、array 和 
callable, 
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1. 强制 模式 
下 面 通过 案例 来 理解 强制 模式 的 含义 。 代 码 如 下 : 


<?php 

// 强制 模式 

function sum(int $ints) 
return array_sum(Sints) 7 

1 

print (sum(2, '3', 8.18)); 

br 


上 述 程 序 输出 结果 为 13。 代 码 中 的 '3' 先 转换 为 3，8.18 先 转换 为 整数 8， 然 后 再 进行 相 加 
操作 。 


2. 严格 模式 
下 面 通过 案例 来 理解 严格 模式 的 含义 。 代 码 如 下 : 


<?php 

// 严格 模式 

declare (strict_types=1) 7 
function sum(int Sints) 
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return array sum($ints); 


} 
Print (ommt2r “3 MN 
三 
上 述 程序 由 于 采用 了 严格 模式 ， 所 以 如 果 参 数 中 出 现 不 是 整数 的 类 型 ， 程 序 执行 时 会 报 
错 ， 如 图 3-16 所 示 。 
| 局 hep//ecalhosjcdejcho3/333php 万- 上 | 节 locahost "ey 
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图 3-16 错误 提示 信息 


在 PHP 7 中 ， 用 户 可 以 声明 函数 返回 值 的 类 型 。 可 以 声明 的 返回 类 型 包括 int、float、 
bool、 string、 interfaces、array 和 callable。 
下 面 通过 案例 来 学 习 PHP 7 如 何 声明 函数 返回 值 的 类 型 。 代 码 如 下 : 


<?php 
declare(strict types=1); 


= 口 x 
:|@ hp/ocaho.. D - © | @ iccalhost 
4 因 各， 宣 看 V】 必 记 A】 工具 WD(H) 


60 


function returnIntValue (int $value): int | 
t 9 








return $value; 


} 





print (returnIntValue (60)); 
全 之 


程序 执行 结果 如 图 3-17 所 示 。 图 3-17 ”声明 函数 返回 值 的 类 型 
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3.7 ”使 用 运算 符 


PHP 包含 3 种 类 型 的 运算 符 。 一 元 运算 符 、 二 元 运算 符 和 三 元 运算 符 。 一 元 运算 符 用 在 
一 个 操作 数 之 前 ， 二 元 运算 符 用 在 两 个 操作 数 之 间 ， 三 元 运算 符 用 在 三 个 操作 数 之 间 。 


3.7.1 算术 运算 符 
算术 运算 符 是 最 简单 也 是 最 常用 的 运算 符 。 常 见 的 算术 运算 符 如 表 3-1 所 示 。 


[ 























第 
&w 
表 3-1 算术 运算 符 章 
运算 符 名 称 喜 
+ 加 法 运算 四 
减法 运算 名 
乘法 运算 了 
/ 除法 运算 | 
% 取 余 运 算 
证 累加 运算 
和 累 减 运算 
【 例 3.16】 使 用 算术 运算 符 (示例 文件 ch03\3.16.php)。 
<?php 
$a=13; 
$b=2; 
echo $a "t+" Sb"="? 
echo $at+$b."<br>"; 
echo $a."-".$b."="; 
echo $a-$b."<br>"; 
echo $a."*".$b."="; 
echo $a*$b."<br>"; 
echo $a."/".$b."="; 
echo $a/$b."<br>"; - 0 x 
SEE 





echo $agS$b ."<br>"7 
echo $a."+t+"."="; 
echo $at++."<br>"; 
和 





13+2=15 A 








echo $a--."<br>"; 13++=13 

?> 14--=14 MY 

程序 运行 结果 如 图 3-18 所 示 。 5 
-| 除了 数值 可 以 进行 自 增 运 算 外 ， 字 符 也 可 以 进行 自 增 运算 操作 。 例 如 b++， 结 果 


Ne 将 等 于 c。 
3.7.2 ”字符 串 连 接 符 


字符 运算 符 “.” 把 两 个 字符 串 连接 起 来 ， 变 成 一 个 字符 串 。 如 果 变 量 是 整 型 或 浮 点 型 ， 
PHP 也 会 自动 把 它们 转换 为 字符 串 输出 。 
【 例 3.17】 使 用 字符 串 连 接 符 (示例 文件 ch03\3.17.php) 


<?php 
$a = "等 闲 识 得 东风 面 ， 万 紫 千 红 总 是 春 。"; 


$b = 10; 
echo "我 读 了 " .$b." 遍 : ".$a 
> 
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程序 运行 结果 如 图 3-19 所 示 。 





- 0O x 
OB-|@ htpi//ocalho. P - ©|| Blocalhost 
文件 (| 篇 壤 {E) 可 看 V)】 收 营 天 A 工具 (1) 帮助 (H) 


我 读 了 10 饥 ， 等 闲 识 得 东风 面 ， 万 紫 干 红 总 是 春 。 
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图 3-19 使 用 字符 串 连接 符 





3.7.3 ”赋值 运算 符 


赋值 运算 符 的 作用 是 把 一 定 的 数值 加 载 给 特定 的 变量 。 
赋值 运算 符 的 具体 含义 如 表 3-2 所 示 。 





表 3-2 赋值 运算 符 
运算 符 名 称 
= 将 右边 的 值 赋 给 左边 的 变量 
二 将 左边 的 值 加 上 右边 的 值 ， 赋 给 左边 的 变量 
二 将 左边 的 值 减 去 右边 的 值 ， 赋 给 左边 的 变量 
二 将 左边 的 值 乘 以 右边 的 值 ， 赋 给 左边 的 变量 
三 将 左边 的 值 除 以 右边 的 值 ， 赋 给 左边 的 变量 
= 将 左边 的 字符 串 连 接 到 右边 
%= 将 左边 的 值 对 右边 的 值 取 余数 ， 赋 给 左边 的 变量 


例如 ，$a-=$b 等 价 于 $a=$a-$b， 其 他 赋值 运算 符 与 之 类 似 。 从 表 3-2 可 以 看 出 ， 赋 值 运 
算 符 可 以 使 程序 更 加 简练 ， 从 而 提高 执行 效率 。 


3.7.4 比较 运算 符 


比较 运算 符 用 来 比较 其 两 端 数 值 的 大 小 。 比 较 运算 符 的 具体 含义 如 表 3-3 所 示 。 
表 3-3 比较 运算 符 


名 称 
相等 
不 相等 
大 于 
小 于 
大 于 等 于 























续 表 
运算 符 名 称 

< 小 于 等 于 

= 精确 等 于 (类 型 ) 

一 不 精确 等 于 





其 中 ，= 一 和 ! 一 需要 特别 注意 。$b=--$c 表示 $b 和 Sc 不 只 是 数值 上 相等 ， 而 且 两 者 的 类 
型 也 一 样 ，$b! 一 $c 表示 $b 和 $c 有 可 能 是 数值 不 等 ， 也 可 能 是 类 型 不 同 。 
【 例 3.18】 使 用 比较 运算 符 ( 示 例文 件 ch03\3.18.php)。 


<R2BHP 

$value="15"; 

echo "\$value = \"$value\""; 
echo "<br>\$value==15: "7 


济 广 计 膨 号 dHd 一 一 妆 站 并 局 主导 蜗 册 人 由 鱼 


var_dump ($value==15); // 结 果 为 :bool (true) 
echo "<br>\$value==true: "7 

var_dump ($value==true); // 结 果 为 :bool (true) 
echo "<br>\$value!=null: "7 

var_dump ($value!=null1); // 结 果 为 :bool (true) 
echo "<br>\$value==false: "7 

var_dump ($value==false); // 结 果 为 :bool (false) 





echo "<br>\$value 
var_dump ($value===100); // 结 果 为 :bool (false) 











echo "<br>\$value===true: "7 
var_dump ($value===true); // 结 果 为 :bool (true) 
echo eber>(LON220 == Se 
var_dump (10/2.0 !==5); // 结 果 为 :bool (true) 


县 区 


程序 运行 结果 如 图 3-20 所 示 。 











- oO x 
全 mpWocaesycodeyrosaap DO] iocalhoet 

x) NE) HBV WRN IRT WO 

Svalue ="15" a 

Svahue 一 15 

Cc: \wamp\mm\ code\ch03\3. 16. Php: :boolean true 

Svalue 一 tme: 

Er\wamp lcode\cho3\3.18. Ehp:7 :boolear true 

Svalue!=null: 

Ernamp lcode\chos\3.18. Fhp: :toolear true 

Svalue—false- 

E+ \wamp\wns\ code\cho3\3.18. Ehp:11:boolean Ealse 

Svalue 一 100: 

aaampVwraNcode\chns\a-18.Fhpz13xboolean false v 

2% 








3-20 ”使 用 比较 运算 符 
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3.7.5 “逻辑 运算 符 


一 个 编程 语言 最 重要 的 功能 之 一 就 是 进行 逻辑 判断 和 运算 。 比 如 逻辑 和 、 侵 辑 或 、 逻 辑 
非 。 逻 辑 运算 符 的 含义 如 表 3-4 所 示 。 














表 3-4 远 辑 运算 符 
运算 符 名 称 
&&、AND 逻辑 和 
| OR 逻辑 或 
!、NOT 逻辑 非 
XOR 逻辑 异 或 





3.7.6” 按 位 运算 符 
按 位 运算 符 是 把 整数 以 “位 ”为 单位 进行 处 理 。 按 位 运算 符 的 含义 如 表 3-5 所 示 。 
表 3-5 按 位 运算 符 


按 位 或 





3.7.7 ”否定 控制 运算 符 
否定 控制 运算 符 用 在 操作 数 之 前 ， 用 于 对 操作 数 真 假 的 判断 。 否 定 控制 运算 符 的 含义 如 
表 3-6 所 示 。 
表 3-6 否定 控制 运算 符 


名 称 
逻辑 非 
按 位 非 











3.7.8 ”错误 控制 运算 符 


错误 控制 运算 符 是 用 @ 来 表示 的 ， 在 一 个 操作 数 之 前 使 用 ， 该 运算 符 用 来 屏蔽 错误 信息 
的 生成 。 


3.7.9 三 元 运算 符 
三 元 运算 符 “? :” 是 作用 在 三 个 操作 数 之 间 的 。 其 语法 格式 如 下 : 


(expI1) ? (expr2) : (expr3) 


如 果 表达 式 exprl 为 真 ， 则 返回 expr2 的 值 ， 如 果 表 达 式 exprl 为 假 ， 则 返回 expr3。 从 
PHP 5.3 开始 ， 可 以 省 略 expr2， 表 达 式 为 (exprl) ?: (expr3)， 如 果 表 达 式 exprl 为 真 ， 则 返回 
exprl 的 值 ， 如 果 表 达 式 exprl 为 假 ， 则 返回 expr3。 例 如 以 下 代码 : 

<?php 

$aa = ' 春 花 秋 月 何 时 了 ， 往 事 知 多 少 '; 

$bb = $aa ?: ' 没 有 古诗 内 容 '; 

$cc = $aa ? ' 这 里 输出 古诗 的 内 容 ': ' 没 有 古诗 内 容 '; 

echo $bb; 

echo $cc; 

2> 


程序 运行 结果 如 图 3-21 所 示 。 








加 "| 全 tpi/localho.. P ~ © | @localhost 
文件 (站 纺 坟 (E) 。 豆 看 (V) 收藏 灾 (A) 工具 (T) 才 助 (H) 


春花 秋月 何 时 了 ， 往 事 知 多 少 。 这 里 输出 古诗 的 内 容 。 
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图 3-21 使 用 三 元 运算 符 
3.7.10 ”运算 符 的 优先 级 和 结合 规则 


运算 符 的 优先 级 和 结合 规则 其 实 与 正常 的 数学 运算 符 的 规则 十 分 相似 。 

(1) 加 减 乘除 的 先后 顺序 与 数学 运算 中 的 完全 一 致 。 

(2) 对 于 括号 ， 则 先 运行 括号 内 ， 再 运行 括号 外 。 

(3) 对 于 赋值 ， 则 由 右 向 左 运行 ， 也 就 是 依次 从 右边 向 左边 的 变量 进行 赋值 。 


3.8 PHP 7 的 新 变化 一 一 合并 运算 符 和 组 合 运 算 符 


PHP 7 新 增加 的 合并 运算 符 (2?) 用 于 判断 变量 是 否 存 在 且 值 不 为 NULL， 如 果 是 ， 它 就 会 


返回 自身 的 值 ， 否 则 返回 它 的 第 二 个 操作 数 。 
语法 格式 如 下 : 


(exprl) ? ? (expr2) 





六 琐 着 肤 否 dHd 一 一己 站 痊 忆 主 条 遇 翰 8 涛 鲁 
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如 果 表 达 式 exprl 为 真 ， 则 返回 exprl 的 值 ， 如果 表达 式 exprl 为 假 ， 则 返回 expr2。 例 


如 以 下 代码 : 
<?php 
$aa = ' 众 鸟 高 飞 尽 ， 孤 云 独 去 闲 。' 
$bb = $aa ?? ' 没 有 古诗 内 容 '; 


echo $bb; 
本 必 


程序 运行 结果 如 图 3-22 所 示 。 


一 口 x 
5)[®-|@ hp//ocaho.. P ™ © | @ localhost 
交 件 (站 ”篇 加 (E) 查看 (V) 。 收 臣 夫 (A) 工具 Mm 帮助 IH) 











众 鸟 高 飞 尽 ， 孤 云 独 去 闲 。 
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PHP 7 新 增加 的 组 合 运算 符 ， 用 于 比较 两 个 表达 式 图 3.22 使 用 合并 运算 符 
$a 和 $b， 如 果 $a 小 于 、 等 于 或 大 于 $b 时 ， 它 分 别 
返回 -1、0 或 1。 例如 以 下 代码 : 


<?php 

// 整 型 比较 

echo( 5 <=> 5);echo "<br>"; 
echo( 5 <=> 6);echo "<br>"7 
echo( 6 <=> 5);echo "<br>"; 


// 浮 点 型 比较 @ [OT TT 


echo( 5.6 <=> 5.6);echo "<br>"; 纺 缠 (E) 可 看 Cy| 。 收 荐 关 A) 工具 (Tm 各 助 (H) 
echo( 5.6 <=> 6.6);echo "<br>"; 0 

echo( 6.6 <=> 5.6);echo "<br>"; -1 

echo (PHP_EOL); 











1 
0 

// 字符 串 比较 1 

echo ( "a" <=> "a") echo "<br>"; | 

echol "a <=> Day ecCheiw<brye 1 

echo ( "b" <=> "a");echo "<br>"; 机 125% 


ye 
图 3-23 ”使 用 组 合 运算 符 
程序 运行 结果 如 图 3-23 所 示 。 





3.9 ”PHP 中 的 表达 式 


表达 式 是 表达 一 个 特定 操作 或 动作 的 语句 。 表 达 式 由 操作 数 和 操作 符 组 成 。 
操作 数 可 以 是 变量 ， 也 可 以 是 常量 。 

操作 符 则 体现 了 要 表达 的 各 个 行为 ， 如 逻辑 判断 、 赋 值 、 运 算 等 。 

例如 $a=5 就 是 表达 式 ， 而 “$a=5;” 则 为 语句 。 另 外 ， 表 达 式 也 有 值 ， 如 表达 式 $a=1 的 


值 为 1。 
Se 在 PHP 代码 中 ， 使 用 “;” 号 来 区 分 表达 式 和 语句 ， 即 一 个 表达 式 和 一 个 分 号 组 
外 成 一 条 PHP 语句 。 在 编写 代码 时 ， 应 特别 注意 表达 式 后 面 的 “;”， 不 要 漏 写 或 写 


错 ， 否 则 会 提示 语法 错误 。 


3.10 创建 多 维 数 组 


前 面 讲述 了 如 何 创建 一 维 数组 ， 下 面 讲述 如 何 创建 多 维 数组 。 多 维 数组 和 一 维 数组 的 区 
别 是 有 两 个 或 多 个 下 标 ， 但 它们 的 用 法 基本 相似 。 

下 面 给 出 创建 二 维 数组 的 例子 。 

【 例 3.19】 创 建 二 维 数组 (示例 文件 ch03\3.19.php)。 


<?php 

$arr[0] [0] = "月 黑 雁 飞 高 "; 
$arr[0] [1] =" 单 于 夜 授 逃 "; 
$arr[1] [0] = "和 欲 将 轻骑 逐 "; 


$arr[1] [1] = "大 雪 满 弓 刀 "7 

for ($i=0; $Si<count (Sarr); $i++) 
for ($k=0; $k<count ($arr[$i]); $k++) 
t 

$arrl = each ($arr[$i]); 

echo "$arrl[value]<br>"; 

. 


> 


程序 运行 结果 如 图 3-24 所 示 。 


讲 玉 诗 膝 否 dHd 一 冯 站 淮 书 宇 朵 钊 翰 8 洲 国 








- OO x 
加 "| 居 nepyecaho- P - © | @localhost 
文件 从 久 纺 (E) 下 看 W) 收藏 交 (A) 工具 (T) 帮助 (H) 


月 黑 雁 飞 高 
单 于 夜 通 逃 
大 雪 满 号 刀 
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图 3-24 创建 二 维 数组 


3.11 疑难 解 惑 


疑问 1: 如 何 灵 活 运用 命名 空间 (namespace)? 


答 : 命名 空间 (namespaces) 作 为 一 个 比较 宽泛 的 概念 ， 可 以 理解 为 用 来 封装 各 个 项 目的 手 
段 。 比 如 ， 文 件 系 统 中 不 同文 件 夹 路 径 中 的 两 个 文件 的 文件 名 可 以 完全 相同 ， 但 由 于 是 在 不 
同 的 文件 夹 中 ， 所 以 是 两 个 完全 不 同 的 文件 。 

PHP 的 命名 空间 也 是 这 样 一 个 概念 。 它 主要 用 于 在 “类 的 命名 ”“ 函 数 命 名 ”及 “常量 
命名 ”中 避免 代码 冲突 和 在 命名 空间 下 管理 变量 名 和 常量 名 。 
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命名 空间 使 用 namespace 关键 字 在 文件 头 部 定义 。 例 如 : 


<?php 

namespace 2ndbuilding\number24; 
class room{} 

Sroom = new _NAMESPACE_ .room; 
2 


命名 空间 还 可 以 拥有 子 空 间 ， 它 们 组 合 起 来 ， 就 像 文 件 夹 的 路 径 一 样 。 对 于 命名 空间 的 
使 用 ， 可 以 通过 内 置 变量 NAMESPACE 来 使 用 命名 空间 及 其 子 空间 。 


疑问 2: 如 何 快速 区 分 常量 和 变量 ? 


答 : 常量 和 变量 的 明显 区 别 如 下 。 

(1) 常量 前 面 没有 美元 符号 ($)。 

(2) 常量 只 能 用 define0 函 数 定义 ， 而 不 能 通过 赋值 语句 定义 。 

(3) 常量 可 以 不 用 理会 变量 范围 的 规则 ， 可 以 在 任何 地 方 定义 和 访问 。 
(4) 常量 一 旦 定义 就 不 能 被 重新 定义 或 者 取消 定义 。 

(5) 常量 的 值 只 能 是 标量 。 

疑问 3: PHP 中 常见 的 输出 方式 有 几 种 ? 


答 : 在 PHP 中 ， 常 见 的 输出 语句 如 下 。 

(1) echo 语句 : 可 以 一 次 输出 多 个 值 ， 多 个 值 之 间 用 逗号 分 隔 。 

(2) print 语句 : 只 允许 输出 一 个 字符 串 。 

(3) print_r0 函 数 : 可 以 把 字符 串 和 数字 简单 地 打印 出 来 ， 而 数组 则 以 括 起 来 的 键 和 值 的 
列表 形式 显示 ， 并 以 Array 开头 。 但 print_r0 输 出 布尔 值 和 NULL 的 结果 没有 意义 ， 因 为 都 是 
打印 Mn"。 因 此 ， 用 var_dumpO 函 数 更 适合 调试 。 

var_dump(O 函 数 : 判断 一 个 变量 的 类 型 与 长 度 ， 并 输出 变量 的 数值 ， 如 果 变 量 有 值 输出 的 
是 变量 的 值 并 回 返 数 据 类 型 。 此 函数 显示 关于 一 个 或 多 个 表达 式 的 结构 信息 ， 包 括 表达 式 的 
类 型 与 值 。 





第 4 章 
实现 定制 功能 
一 函数 的 应 用 


在 PHP 7 语言 中 ， 提 供 了 超过 1000 个 以 上 的 函数 。 这 些 函 数 在 任何 需要 的 时 
候 都 可 以 被 随时 调用 ， 从 而 提高 了 开发 软件 的 效率 ， 也 提高 了 程序 的 重用 性 和 可 靠 
性 ， 使 软件 维护 起 来 更 加 方便 。 这 也 是 函数 在 PHP 中 的 最 大 魅力 。 本 章 重 点 学 习 
内 置 函 数 、 自 定义 函数 和 包含 文件 等 知识 。 
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4.1 认识 范 数 


函数 的 英文 为 fanction， 这 个 词 也 是 功能 的 意思 。 顾 名 思 义 ， 使 用 函数 就 是 要 在 编程 过 程 
中 实现 一 定 的 功能 ， 也 就 是 通过 一 段 代码 块 来 实现 一 定 的 功能 。 比 如 ， 通 过 一 定 的 功能 记录 
下 酒店 客人 的 个 人 信息 ， 每 到 他 生日 的 时 候 自动 给 他 发 送 祝福 E-mail， 并 且 这 个 发 信 “ 功 
能 ”可 以 重用 ， 将 来 在 某 个 客户 的 结婚 纪念 日 也 使 用 这 个 功能 给 他 发 送 祝 福 E-mail。 可 见 ， 
函数 就 是 实现 一 定 功能 的 一 段 特定 的 代码 。 

实际 上 ， 前 面 我 们 早已 使 用 过 函数 了 。 例 如 ， 用 define0 函 数 定义 一 个 常量 。 如 果 现在 再 
写 一 个 程序 ， 则 同样 可 以 调用 define0 函 数 。 





4.2 内 置 函 数 


PHP 提供 了 大 量 内 置 函数 ， 以 方便 程序 员 直 接 使 用 。 常 见 的 内 置 函 数 包 括 数学 函数 、 变 
量 函 数 、 字 符 串 函数 、 时 间 和 日 期 函数 等 。 由 于 字符 串 函 数 、 时 间 和 日 期 函数 在 后 面 的 章节 
中 将 详细 介绍 ， 所 以 本 节 主 要 讲述 内 置 的 数学 函数 和 变量 函数 。 


4.2.1 数学 函数 


数学 函数 主要 用 于 实现 数学 上 的 常用 运算 ， 主 要 处 理 程序 中 int 和 float 类 型 的 数据 。 
1. 随机 函数 rand() 
随机 函数 rand0 的 语法 格式 如 下 : 


int rand([int min,int max]) 


返回 min 到 max 之 间 随 机 的 整数 。 如 果 min 和 max 参数 忽略 ， 则 返回 0 到 RAND_MAX 


之 间 的 随机 整数 。 
下 面 以 调用 数学 函数 为 例 进 行 讲解 。 
【 例 4.1】 使 用 rand0 函 数 (示例 文件 ch04\4.1.php)。 
<?php 
echo rand ()."<br/>"; // 返 回 随机 整数 
echo rand (100,200); // 产 生 一 个 100~200 之 间 的 随机 整数 


> 


程序 运行 结果 如 图 4-1 所 示 。 每 刷新 一 次 页 面 ， 显 示 结果 都 不 相同 。 


- DO x 
@ 人 @ htpylocalhosycode 只 - © | Blocalhost 
文件 (月 忽 强 (E) 得 看 (V) 。 收 莹 夫 (A) 工具 (T) 帮助 (H) 


14870 
124 
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4-1 使 用 rand() 函 数 





2. 舍 去 法 取 整 函数 floor() 
舍 去 法 取 整 函数 floor 的 语法 格式 如 下 : 


float floor (float value) 


返回 不 大 于 value 的 下 一 个 整数 ， 将 value 的 小 数 部 分 舍 去 取 整 。 
【 例 4.2】 使 用 floor0 函 数 ( 示 例文 件 ch04\4.2.php)。 

<?php 

echo floor (5.66)."<br/>"; // 舍 去 法 取 整 数 


echo floor (8.1234) 7 
Te 


二 区 B 吉 直 主 Ng 半 是 关 国 


程序 运行 结果 如 图 4-2 所 示 。 [ni 


"| hpi/ocalho.. P+ © 





3. 对 浮上 点数 四 使 五 入 的 函数 round() 
四 售 五 入 的 函数 round0 的 语法 格式 如 下 : 


int round (float val,int precision) 


返回 将 val 根据 指定 精度 precision 进行 四 舍 五 入 的 结 图 4-2 使 用 floor() 函 数 
果 。 其 中 precision 可 以 为 负数 或 者 零 (默认 值 )。 
【 例 4.3】 使 用 roundO 函 数 (示例 文件 ch04\4.3.php)。 
<?php 
echo round(5.66)."<br/>"; // 四 合 五 入 法 取 整 数 
echo round(8.12)."<br/>"; 
echo round(8.1234,2)."<br/>"; 


echo round(1234.6,-2)."<br/>"; 
> 


程序 运行 结果 如 图 4-3 所 示 。 





R125% > 








一 口 x 
@ -| epocho. Pp- | locshen 





4.2.2 ”变量 相关 的 函数 eee ey ce an ee 
8 
在 PHP 7 中 ， 针 对 变量 相关 的 函数 比较 多 。 下 面 挑选 I 
比较 常用 的 函数 进行 讲解 。 
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1. 检验 变量 是 否 为 空 的 函数 emptyO 
4-3 使 用 round() 函 数 


bool empty (mixed Var) 


如 果 var 是 非 空 或 非 零 的 值 ， 则 empty0 返 回 FALSE; 如 果 var 为 空 ， 则 返回 TRUE。 
【 例 4.4】 使 用 empty0 函 数 (示例 文件 ch04\4.4.php)。 


<?php 

$a=5; 

$b=" 春 花 秋月 何 时 了 "; 

$c= null; 

$d= 0; 

var_dump (empty ($a) ) ."<br/>"; // 输 出 变量 的 值 和 类 型 
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Var_dump (empty ($b))."<br/>"; 
var_dump (empty ($c))."<br/>"; 
Var_dump (empty ($9))."<br/>"; 
3x: 


程序 运行 结果 如 图 4-4 所 示 。 


- EEE3 
加 -| 碍 papyWocaho- P - 6] 大 Iocalhos x 
交 件 朋 多 昌吉 本 VW) 收工 上 Mm 帮 有 HH 


C: \wanp\wa\code\chO4\4.4.php:4:boclean false 











C: \wanp\wwa\code\ch04\4.4.php:5:boolean false 





C: wanp\wwa\code\ch04\4.4.php:6:boolean TILe 


C: Wwanp\wwa\code\ch04\4.4.php:7:boolean LIUE 
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4-4 使 用 empty() 函 数 
2. 判断 变量 是 否定 义 过 的 函数 isset() 


bool isset ( mixed var [, mixed var [, ...]] ) 


若 变 量 var 不 存在 则 返回 FALSE; 若 变量 存在 且 其 值 为 NULL， 也 返回 FALSE; 若 变 量 
存在 且 值 不 为 NULL， 则 返回 TRUE。 同 时 检查 多 个 变量 时 ， 每 个 变量 被 检查 时 都 返回 TRUE 
时 ， 结 果 才 为 TRUE， 和 否则 结果 为 FALSE。 

【 例 4.5】 使 用 isset0 函 数 (示例 文件 ch04\4.5.php)。 


<?php 

$a=5; 

$b=" 春 花 秋月 何 时 了 "; 

$c= null; 

var_dump (isset ($a))."<br/>"; // 输 出 变量 的 值 和 类 型 
var_dump (isset ($b))."<br/>"; 

var dump (isset ($c))."<br/>"; 

var_dump (isset ($b, $c))."<br/>"; 

> 


程序 运行 结果 如 图 4-5 所 示 。 








sd al 
轩 -| 息 hpyfiocalho-. PD- Cj 七 localhost 
文件 (有 ) ”编辑 (E) 查看 (V) 收藏 去 (A) 工具 (T) 帮助 (H) 


x | X 








C:\wamp\wa\code\ch04\4.5.php:5:boolean true 
C:\wamp\wa\code\ch04\4.5.php:6:boolean true 
C:\wamp\wa\code\ch04\4.5.php:7:boolean false 


C:\wamp\wa\code\ch04\4.5.php:8:boolean false 
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4-5 使 用 isset() 函 数 





4.3 自 定义 函数 
PHP 不 仅 提供 了 大 量 的 内 置 函数 ， 用 户 还 可 以 自 定义 函数 。 


4.3.1 定义 和 调用 范 数 
在 更 多 的 情况 下 ， 程 序 员 面 对 的 是 自 定义 函数 。 其 语法 格式 如 下 : 


function name of function(paraml, param2, ...){ 
statement; 


; 
其 中 ，name_of function 是 函数 名 ，paraml1、param2 是 参数 ， 而 statement 是 函数 的 具体 


滞 则 呈 洲 加 一 下 叶 吾 训 尖 六 _ 协 y 浇 轿 





内 容 。 
下 面 以 自 定义 和 调用 函数 为 例 进行 讲解 。 
【 例 4.6】 定 义 和 调 用 函数 (示例 文件 ch04\4.6.php)。 


<?php 

function gushi ($name){ // 自 定义 函数 gushi 

return $name."<br/>"." 胜 日 寻 芳 泗水 滨 ， 无 边 光景 一 时 新 。" . "<br/>". "等闲 识 得 东风 面 ， 万 
紫 千 红 总 是 春 。"; 

echo gushi (' 春 日 ') ; // 调 用 函数 gushi 











Sa 
程序 运行 结果 如 图 4-6 所 示 。 -一 六 
Ej rpviccelhe.. P - ¢ |@ ccahost 
【案例 训 析 】 A 
春日 


值得 一 提 的 是 ， 此 函数 是 以 值 的 形式 返回 的 。 也 就 是 。。 先天 时 
说 ，return 语句 返回 值 时 ， 创 建 了 一 个 值 的 拷贝 ， 并 把 它 返 
回 给 使 用 此 函数 的 命令 或 函数 ， 在 这 里 是 echo 命令 。 





4.3.2 向 函数 传递 参数 值 图 4-6 定义 和 调用 函数 
由 于 函数 是 一 段 封 闭 的 程序 ， 很 多 时 候 ， 程 序 员 都 需要 向 函数 内 传递 一 些 数据 ， 来 进行 





操作 。 
可 以 接受 传 入 参数 的 函数 定义 形式 如 下 : 
function 函数 名 称 (参数 1， 参数 2) { 
算法 描述 ， 其 中 使 用 参数 1 和 参数 2; 
} 
下 面 通过 案例 来 学 习 如 何 向 函数 传递 参数 值 。 
【 例 4.7】 向 函数 传递 参数 值 (示例 文件 ch04\4.7.php)。 


<?php 
function coul($a, $b){ // 定 义 函 数 
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$c= $a*s$b; 
echo "计算 结果 为 : " .$c; 
} 














Sa // 定 义 全 局 变量 

$b = 8; 

cou ($a, $b); // 调 用 函数 cou () 

echo "<br/>"; 

cou(5,8); // 调 用 函数 cou () 

2> 

程序 运行 结果 如 图 4-7 所 示 。 i 

多 Ripylocahesycede Pv O |[ localhost 

【案例 剖析 】 © 第 吉 (E) 二 看 V】 收藏 A) nt 

(1) 以 这 种 方式 传递 参数 值 的 方法 就 是 向 函数 传递 参半 入 名 | 
数值 。 R125% 





(2) 其 中 fonction cou ($a,$b) 们 定义 了 函数 和 参数 。 
(3) 不 管 是 通过 变量 $a 和 $b 向 函数 内 传递 参数 值 ， 还 
是 像 cou(5,8) 这 样 直接 传递 参数 值 ， 效 果 都 是 一 样 的 。 


4.3.3 向 函数 传递 参数 引用 


向 函数 传递 参数 引用 ， 其 实 就 是 向 函数 传递 变量 引用 。 参 数 引用 一 定 是 变量 引用 ， 静 态 
数值 是 没有 引用 一 说 的 。 变 量 引用 其 实 就 是 对 变量 名 的 使 用 ， 即 是 对 变量 位 置 的 使 用 。 

下 面 通过 案例 来 学 习 。 

【 例 4.8】 向 函数 传递 参数 引用 (示例 文件 ch04\4.8.php)。 

<?php 

$a = 300; 

$b = 507 

function totall(g$a, $b){ 

$a = $b + $b; 

echo " 求 和 运算 的 结果 为 : $b"; 

上 

total ($a, Sb) 

echo "<br/>"; 

total ($a, $b); 

党 次 


程序 运行 结果 如 图 4-8 所 示 。 30 
【案例 剖析 】 | rr 


文件 ff) 短 误 E) 各 者 (V) 收藏 实 (A) 工具) #4 
(1) 以 这 种 方式 传递 参数 值 的 方法 就 是 向 函数 传递 参数 求生 得 引 仙 为 280 
引用 。 使 用 & 符 号 表示 参数 引用 。 
(2) 其 中 function total(&$a, $b) 们 定义 了 函数 、 参 数 和 参数 
引用 。 变 量 $a 是 以 参数 引用 的 方式 进入 函数 的 。 当 函数 的 运行 。 一 一 “一 
结果 改变 了 变量 $a 引用 的 时 候 ， 在 函数 外 的 变量 $a 的 值 也 发 生 ”图 4-8 ”向 函数 传递 参数 引用 
了 改变 。 也 就 是 函数 改变 了 外 部 变量 的 值 。 


4-7 ”向 函数 传递 参数 值 





4.3.4 从 函数 中 返回 值 


以 上 的 一 些 例子 ， 都 是 把 函数 运算 完成 的 值 直接 打印 出 来 。 但 是 ， 在 很 多 情况 下 ， 程 序 
并 不 需要 直接 把 结果 打印 出 来 ， 而 是 仅仅 给 出 结果 ， 并 且 把 结果 传递 给 调用 这 个 函数 的 程 
序 ， 为 其 所 用 。 

这 里 需要 用 到 return 关键 字 。 

【 例 4.9】 从 函数 中 返回 值 (示例 文件 ch04\4.9.php)。 


<?php 

function lian($a,$b){ 

return $a.s$b; 

i 

$a = "应 怜 展 齿 印 苍 苦 ， 小 扣 柴 靡 久 不 开 。"; 
$b = "春色 满 园 关 不 住 ， 一 枝 红 杏 出 墙 来 。"; 
echo lian($a, $b); 

2 


汪 加 一 下 时 可 当 六 _ 志 v 牙关 














程序 运行 结果 如 图 4-9 所 示 。 

ox 
@ [TREE 下 让 放生 g 

六 件 们 ” 测 霹 ()。 豆 看 (V)】 履 直 六 (A) 工 如 本 入) 
应 怜 展 堵 印 苍 蔡 ， 小 扣 柴 雇 久 不 开 。 春 色 满 园 关 不 住 。 一 枝 红 理 出 堵 米 。 
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图 4-9 从 函数 中 返回 值 
【案例 剖析 】 


(1) 在 函数 function lian($a,$b) 的 算法 中 ， 直 接 使 用 retum 把 运算 的 值 返回 给 调用 此 函数 
的 程序 。 

(2) 其 中 echo lian($a,$b) 语 句 调用 了 此 函数 ，lian0 把 运算 结果 值 返回 给 了 echo 语句 ， 才 
有 上 面 的 显示 。 当 然 ， 这 里 也 可 以 不 用 echo 来 处 理 返 回 值 ， 也 可 以 对 它 进 行 其 他 处 理 ， 比 如 
赋值 给 变量 等 。 


4.3.5 引用 函数 


不 管 是 PHP 中 的 内 置 函数 ， 还 是 程序 员 在 程序 中 的 自 定义 函数 ， 都 可 以 直接 简单 地 通过 
函数 名 调用 。 但 是 在 操作 过 程 中 也 有 些 不 同 ， 大 致 分 为 以 下 3 种 情况 。 

(1) 如 果 是 PHP 的 内 置 函数 ， 如 date0， 可 以 直接 调用 。 

(2) 如 果 这 个 函数 是 PHP 的 某 个 库 文件 中 的 函数 ， 则 需要 用 include0 或 require0 命 令 把 此 
库 文 件 加 载 ， 然 后 才能 使 用 。 

(3) 如 果 是 自 定义 函数 ， 若 与 引用 程序 在 同一 个 文件 中 ， 则 可 直接 引用 ; 若 此 函数 不 在 当 
前 文件 内 ， 则 需要 用 include0 或 require0 命 令 加 载 。 

对 函数 的 引用 ， 实 质 上 是 对 函数 返回 值 的 引用 。 
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【 例 4.10】 引 用 函数 (示例 文件 ch04\4.10.php)。 


<?php 

function &example ($aa=1){ // 定 义 一 个 函数 ， 别 忘 了 加 & 符 号 
return $aa; // 返 回 参 数 $aa 

} 

$bb = &example ("引用 函数 的 实例 ") ; // 声 明 一 个 函数 的 引用 Sbb 
echo $bb; 

全 > 


程序 运行 结果 如 图 4-10 所 示 。 














【案例 剖析 】 A a 
(1) 本 例 首先 定义 一 个 函数 ， 然 后 变量 $bb 将 引用 函 “下 于 的 雪人 . 
数 ， 最 后 输出 变量 Sbb， 实 质 上 是 $aa 的 值 。 
(2) 与 参数 传递 不 同 ， 使 用 函数 引用 时 ， 定 义 函数 和 el 
引用 函数 都 必须 使 用 & 符 号 ， 表 明 返 回 的 是 一 个 引用 。 
图 4-10 引用 函数 


4.3.6 ”取消 函数 引用 


对 于 不 需要 引用 的 函数 ， 可 以 做 取消 操作 。 取 消 引用 函数 使 用 unset0 函 数 来 完成 ， 目 的 
是 断 开 变量 名 和 变量 内 容 之 间 的 绑 定 ， 此 时 并 没有 销毁 变量 内 容 。 
【 例 4.11】 取 消 函 数 引 用 (示例 文件 ch04\4.11.php)。 


<?php 

$a = 86; // 声 明 一 个 整 型 变量 

$b = &$a; // 声 明 一 个 对 变量 $a 的 引用 $b 
echo "变量 b 的 值 为 : " .$b."<br/>"; // 输 出 引用 $b 

unset ($b); // 取 消 引 用 $b 

echo "变量 b 的 值 为 : ". $b."<br/>"; // 再 次 输出 引用 

echo "变量 a 的 值 为 :". $a; // 输 出 原 变 量 


区 
程序 运行 结果 如 图 4-11 所 示 。 


(Om 
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变量 b 的 值 为 :86 














各 量 b 了 现价 为 : 
变量 a 的 值 为 : 86 


RN 





图 4-11 取消 函数 引用 
【案例 剖析 】 
本 程序 首先 声明 一 个 变量 和 变量 的 引用 ， 输 出 引用 后 取消 引用 ， 再 次 调用 引用 和 原 变 


量 。 从 图 4-11 可 以 看 出 ， 取 消 引 用 后 ， 再 次 调用 引用 Sb， 将 会 提示 警告 信息 ;再 次 调用 原 变 
量 $a， 可 以 发 现 取 消 函 数 引 用 后 ， 对 原 变量 没有 任何 影响 。 





全 


4.3.7 变量 函数 


所 谓 变 量 函 数 ， 是 指 通过 变量 来 访问 的 函数 。 当 变量 后 有 圆 括 号 时 ，PHP 将 自动 寻找 与 


变量 的 值 同名 的 函数 ， 然 后 执行 该 函数 。 


【 例 4.12】 变 量 函 数 ( 示 例文 件 ch04\4.12.php)。 


<?php 
function ff() { // 声明 ff () 函数 
echo “墙角 数 枝 梅 ， 凌 寒 独 自 开 。<br/>'7 
echo “' 遥 知 不 是 雪 ， 为 有 暗 香 来 。<br/>'"7 
} 
function fc($string) {// 声明 fc () 函数 
echo ' 调 用 ff () 函数 !<br/>'; 
echo $string; 
} 
SvarEtte vee // 将 fun 函数 名 赋值 给 变量 
Svar_ff() 7 0 调用 ff () 函数 
Svar ££ = "fe 
$var ff(' 泉 了 于 异 细 沪 ( 有 晴 条 。 的 克 
/* 
调用 ff () 函数 ! 
通过 改变 变量 的 值 ， 实现 调用 其 他 函数 
a 


学 六 


程序 运行 结果 如 图 4-12 所 示 。 


EEC | Bem 
文 站 站 。 罗 笛 日。 二 寿 V】 避 吾 (A] 工具 Mm 大 动 IH) 
墙角 数 枝 悔 ， 凌 寒 独自 开 . 
遥 知 不 是 当 ， 为 有 暗 香 来 。 

调用 ff() 函数 ! 

泉眼 尤 声 惜 细 流 ， 树 阴 照 水 爱 晴 柔 , 





2% ~ 


4-12 ”变量 函数 


4.4 PHP7 的 新 变化 一 一 新 增 intdiv() 函 数 


在 PHP 7 中， 新 增 了 整除 函数 intdiv0。 其 语法 格式 如 下 : 


intdiv(a, b); 


该 函数 返回 值 为 a 除 以 b 的 值 并 取 整 。 
【 例 4.13】 使 用 intdiv0 函 数 (示例 文件 ch04\4.13.php)。 


<?php 

echo intdiv(10, 3)."<br/>"; 
echo, intdiv(6, 3) ."<br/>"> 
echo intdiv(l, 2) "<br/>"; 
?> 








到 同 台海 四 二 号 二 入 当时 由 小国 
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程序 运行 结果 如 图 4-13 所 示 。 
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4-13 使 用 intdiv() 函 数 


4.5 包含 文件 


如 果 想 让 自 定义 的 函数 被 多 个 文件 使 用 ， 可 以 将 自 定义 函数 组 织 到 一 个 或 者 多 个 文件 
中 ， 这 些 收集 函数 定义 的 文件 就 是 用 户 自己 创建 的 PHP 函数 库 。 通 过 使 用 require0 和 include0 
等 函数 可 以 将 函数 库 载 入 到 脚本 程序 中 。 


4.5.1 require 和 include 


require0 和 include0 语 句 不 是 真正 意义 上 的 函数 ， 属 于 语言 结构 。 通 过 include0 和 requireO 
语句 都 可 以 实现 包含 并 运行 指定 文件 。 

(1) require0: 在 脚本 执行 前 读 入 它 包 含 的 文件 ， 通 常 在 文件 的 开头 和 结尾 处 使 用 。 

(2) include0: 在 脚本 读 到 它 的 时 候 才 将 包含 的 文件 读 进 来 ， 通 常 在 流程 控制 的 处 理 区 使 用 。 

require0 和 includeO 语 句 对 于 处 理 失败 方面 是 不 同 的 。 当 文件 读 取 失 败 后 ，require 将 产生 
一 个 致命 错误 ， 而 include 则 产生 一 个 警告 。 可 见 ， 如 果 遇 到 文件 丢失 时 需要 继续 运行 ， 则 使 
用 include， 如 果 想 停止 处 理 页 面 ， 则 使 用 require。 

【 例 4.14】 使 用 include( 示 例文 件 ch04\4.14.php 和 test.php)。 

其 中 ，4.14.php 代码 如 下 : 


<?php 

$aaa = ' 杨 柳 青 青 江水 平 '; // 定 义 一 个 变量 aaa 
$bbb = ' 闻 郎 江上 唱歌 声 ' ; // 定 义 一 个 变量 bbb 
?> 

testphp 代码 如 下 : 

<?php 


echo " $aaa $bbb"; ”// 未 载 入 文件 前 调用 两 个 变量 
include '4.14.php'; 

echo "$aaa $bbb "; // 载 入 文件 后 调用 两 个 变量 
> 


运行 testphp 的 结果 如 图 4-14 所 示 。 从 结果 可 以 看 出 ， 使 用 include 时 ， 虽 然 出 现 了 警 
告 ， 但 是 脚本 程序 仍然 在 运行 。 
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图 4-14 使 用 include 


4.5.2 include_once 和 require_once 


include_once 和 require_once 语句 在 脚本 执行 期 间 包 含 并 运行 指定 文件 ， 作 用 与 include 和 
require 语句 类 似 。 唯 一 的 区 别 是 ， 如 果 该 文件 的 代码 被 包含 了 ， 则 不 会 再 次 包含 ， 只 会 包含 
一 次 ， 从 而 避免 函数 重 定义 以 及 变量 重 赋值 等 问题 。 


4.6 疑难 解 惑 





疑问 1: 什么 是 递归 函数 ? 

答 : 在 PHP 语言 中 ， 函 数 直接 或 间接 调用 函数 本 身 ， 则 该 函数 称 为 递归 函数 。 在 使 用 递 
归 函 数 时 ， 需 要 注意 死 循 环 问题 。 

下 面 通 过 案例 来 了 解 递归 函数 的 使 用 方法 。 

【 例 4.15】 使 用 递归 函数 (示例 文件 ch04\4.15.php)。 


<?php 
function tt($n) { // 声 明 自 定义 函数 
echo $n."<br/>"; // 函数 体内 可 执行 语句 ， 显 示 实 参 值 
if (Sn<0) // 条 件 判 断 是 否 执行 或 终止 递归 动作 
tt (Sn+1) 7 // 开 始 递归 ， 并 给 出 附加 条 件 改变 变量 的 值 ， 防 止 死 循环 
tt (-15) 7 // 执行 递归 函数 
人 


程序 运行 结果 如 图 4-15 所 示 。 
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图 4-15 使 用 递归 函数 
疑问 2: 如 何 销毁 指定 的 变量 ? 
答 : 在 PHP 中 ， 用 户 可 以 通过 unsetO 函 数 销毁 指定 的 变量 ， 还 可 以 同时 销毁 多 个 变量 。 
例如 同时 销毁 变量 a、b 和 ce， 代码 如 下 : 
unset (a,b,c) 


值得 注意 的 是 ， 对 全 局 变量 而 言 ， 如 果 在 函数 内 部 销毁 ， 只 是 在 函数 内 部 起 作用 ， 而 函 
数 调用 结束 后 ， 全 局 变量 依然 存在 并 有 效 。 

疑问 3: 如 何 合理 运用 include_ once0 和 require_onceO? 

答 : include0 和 require0) 函 数 在 其 他 PHP 语句 执行 之 前 运行 ， 引 入 需要 的 语句 并 加 以 执 
行 。 但 是 每 次 运行 包含 此 语句 的 PHP 文件 时 ，include0 和 require0 函 数 都 要 运行 一 次 。 
include0 和 require0 函 数 如 果 在 先前 已 经 运行 过 ， 并 且 引 入 了 相同 的 文件 ， 则 系统 就 会 重复 引 
入 这 个 文件 ， 从 而 产生 错误 。 而 include_ once0 和 require_once0) 函 数 只 是 在 此 次 运行 的 过 程 中 
引入 特定 的 文件 或 代码 ， 但 是 在 引入 之 前 ， 会 先 检查 所 需 文件 或 者 代码 是 否 已 经 引入 ， 如 果 
已 经 引入 ， 将 不 再 重复 引入 ， 从 而 不 会 造成 冲突 。 


疑问 4: 程序 检查 后 正确 ， 却 显示 Notice: Undefined variable， 为 什么 ? 


答 : PHP 默认 配置 会 报 这 个 错误 ， 就 是 将 警告 在 页 面 上 打印 出 来 ， 虽 然 这 有 利于 暴露 问 
题 ， 但 实际 使 用 过 程 中 会 存在 很 多 问题 。 

通用 的 解决 办 法 是 修改 php.ini 的 配置 ， 需 要 修改 的 参数 如 下 。 

(1) 找到 error reporting =E_ALL， 修 改 为 error reporting =E ALL & ~E NOTICE。 

(2) 找到 register globals = Off， 修 改 为 register_ globals = On。 





第 5 章 

程序 的 执行 方向 

一 程序 控制 
结构 


编程 语言 都 是 由 各 种 程序 结构 组 成 的 ， 常 见 的 有 顺序 结构 、 分 支 结构 和 循环 结 
构 。 理 解 程序 的 结构 ， 对 于 学 习 PHP 非常 重要 。 本 章 主要 介绍 PHP 语言 中 的 程序 
结构 的 使 用 方法 和 技巧 。 
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5.1 ”流程 控制 概述 


流程 控制 ， 也 叫 控制 结构 ， 是 在 一 个 应 用 中 用 来 定义 执行 程序 流程 的 程序 。 它 决定 了 某 
个 程序 段 是 否 会 被 执行 和 执行 多 少 次 。 

PHP 中 的 控制 语句 分 为 3 类 : 顺序 控制 语句 、 条 件 控制 语句 和 循环 控制 语句 。 其 中 顺序 
控制 语句 是 从 上 到 下 依次 执行 的 ， 这 种 结构 没有 分 支 和 循环 ， 是 PHP 程序 中 最 简单 的 结构 。 
下 面 主要 讲述 条 件 控制 结构 和 循环 控制 结构 。 


5.2 ”条件 控制 结构 


条 件 控制 结构 中 包含 两 个 主要 的 语句 : 一 是 站 语句 ， 二 是 switch 语句 。 





5.2.1 单一 条 件 分 支 结构 (if 语句 ) 


让 语句 是 最 为 常见 的 条 件 控制 语句 。 它 的 语法 格式 如 下 : 条 
if (条 件 判 断 语句 ) { 
执行 语句 ; 


| 
电 \ 膏 小 





程序 块 





| 

这 种 形式 只 是 对 一 个 条 件 进 行 判断 。 如 果 条 件 成 立 ， 则 执行 
命令 语句 ， 否 则 不 执行 。 

站 语句 的 控制 流程 如 图 5-1 所 示 。 

【 例 5.1】 使 用 让 语句 (示例 文件 ch05\5.1.php)。 





| 








府 
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图 5-1 j 半 语句 的 控制 流程 


<?php 

$a = rand(1,200); // 使 用 rand () 函数 生成 一 个 随机 数 
echo "变量 a 值 为 : " .$a."<br/>"; 

if ($a>50){ // 判 断 变量 a 是 否 大 于 50 

echo "变量 a 大 于 50"; // 输 出 信息 ， 变 量 a 的 值 大 于 50 


> 


运行 后 刷新 页 面 ， 结 果 如 图 5-2 所 示 。 
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图 5-2 使 用 if 语 句 





50， 
， 则 刷新 页 面 即 可 。 


【 委 例 剖析 】 


(1) 此 案例 首先 使 用 rand0 函 数 随 机 生成 一 个 整数 Sa。 然 后 判断 这 个 随机 整数 是 不 是 大 于 
如 果 是 ， 则 输出 上 述 结果 ; 如 果 不 是 ， 则 不 输出 任何 内 容 。 所 以 如 果 页 面 内 容 显示 为 


(2) rand0 函 数 返 回 随机 整数 。 其 语法 格式 如 下 : 
rand (min,max) 


此 函数 主要 是 返回 min 和 max 之 间 的 一 个 随机 整数 。 如 果 没 有 提供 可 选 参数 min 和 


max，rand() 将 返回 0 到 RAND _MAX 之 间 的 随机 整数 。 
5.2.2 ”双向 条 件 分 支 结构 (if...else 语句 ) 





如 果 是 非 此 即 彼 的 条 件 判断 ， 可 以 使 用 让 ..else 语句 。 它 的 语法 格式 如 下 : 
if (条 件 判断 语句 ) { 
执行 语句 A; 
}elsel{ 
执行 语句 B; 
} 
这 种 结构 形式 首先 判断 条 件 是 否 为 真 ， 如 果 为 真 ， 则 执行 语句 A， 和 否则 执行 语句 B。 
f...else 语句 的 控制 流程 如 图 5-3 所 示 。 
【 例 5.2】 使 用 直 ..else 语句 (示例 文件 ch05\5.2.php)。 


<?php 


$a = rand(1,200); // 使 用 rand () 函数 生成 一 个 随机 数 
echo "变量 a 值 为 : " .$a."<br/>"; 

if ($a>50){ // 判 断 变量 a 是 否 大 于 50 

echo "变量 a 大 于 50"; // 输 出 信息 ， 变 量 a 的 值 大 于 50 
}elsel // 和 否则 输出 变量 小 于 或 等 于 50 


echo "变量 a 小 于 或 等 于 50"; 
} 


2> 


程序 运行 结果 如 图 5-4 所 示 。 
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Ql25% 
5-3 if..else 的 控制 流程 5-4 ”使 用 if...else 语句 
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5.2.3 ”多 向 条 件 分 支 结构 (elseif 语句 ) 


在 条 件 控制 结构 中 ， 有 时 会 出 现 多 于 两 种 的 选择 ， 此 时 可 以 使 用 elseif 语句 。 它 的 语法 格 
式 如 下 : 


和 pi 
命令 执 

ein i { 
命令 执 

: 





else{ 
命令 执行 语句 ; 
) 


elseif 语 句 的 控制 流程 如 图 5-5 所 示 。 
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图 5-5 elseif 语句 的 控制 流程 
【 例 5.3】 使 用 else 站 语句 (示例 文件 ch05\5.3.php)。 


<?php 


$score = 75; // 设 置 成 绩 变量 $score 

if ($score >= 0 and $score <= 60){ // 判 断 成 绩 变量 是 否 在 0~60 之 间 
echo "您 的 成 绩 为 差 "; // 如 果 是 ， 说 明成 绩 为 差 

jelseif($score > 60 and $score <= 80){ // 否 则 判断 成 绩 变量 是 否 在 61~80 之 间 
echo "您 的 成 绩 为 中 等 "; // 如 果 是 ， 说 明成 绩 为 中 等 

}elset // 如 果 两 个 判断 都 是 false， 则 输出 默认 值 
echo "您 的 成 绩 为 优等 "; // 说 明成 绩 为 优等 


| 
> 


程序 运行 结果 如 图 5-6 所 示 。 





@ WH-|@ napyilocalho- P - 上 6 | Blocalhost 
文件 (月 ”篇 沪 (E) 。 坦 看 (V) 收藏 夫 (A) ”工具 (D) 帮助 (H) 


您 的 成 绩 为 中 等 











2 
| 





大 125% ~ 





图 5-6 使 用 elseif 语 句 


5.2.4 多 向 条 件 分 支 结 构 (switch 语句 ) 


switch 语句 的 结构 给 出 不 同情 况 下 可 能 执行 的 程序 块 ， 条 件 满足 哪个 程序 块 ， 就 执行 哪 
个 。 它 的 语法 格式 如 下 : 


switch (条 件 判断 语句 ) { 
case 判断 结果 为 a: 
执行 语句 17 

break; 

case 判断 结果 为 b: 
执行 语句 2; 


break; 


音 上 震中 汕 济 甫 一 一 可 斗 站 迁 玻 河 甫 4s 当 国 


Be 
执行 语句 n; 


若 “ 条 件 判断 语句 ”的 结果 符合 哪个 可 能 的 “判断 结果 ”， 就 执行 其 对 应 的 “执行 语 
句 ”。 如 果 都 不 符合 ， 则 执行 default 对 应 的 默认 “执行 语句 n”。 
switch 语句 的 控制 流程 如 图 5-7 所 示 。 
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5-7 ”switch 语句 的 控制 流程 
【 例 5.4】 使 用 switch 语句 (示例 文件 ch05\5.4.php)。 
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Switch ($x) 

{ 

case 1: 

echo "数值 为 1"; 

break; 

case 2: 

echo "数值 为 2"; 

break; 

case 3: 

echo "数值 为 3"; 

break; 

case 4: 

echo "数值 为 4"; 

break; 
ease SE @ [@- pocaho. P - |B locahost 
echo "数值 为 5"; 交 周 们 六 [E] 豆 看 V) 收 吏 大 (A】 工具 [和 有 (H) 
break; 数值 为 3 

default: 

echo "数值 不 在 1~5 之 间 "; 

2% 
2 


程序 运行 结果 如 图 5-8 所 示 。 














图 5-8 使 用 switch 语句 


5.3 ”循环 控制 结构 


循环 控制 语句 中 主要 包括 3 个 语句 ， 即 while 循环 、do...while 循环 和 for 循环 。while 循 
环 在 代码 运行 的 开始 检查 条 件 的 真 假 ， 而 do...while 循环 则 是 在 代码 运行 的 末尾 检查 条 件 的 真 
假 ， 所 以 do...while 循环 至 少 要 运行 一 次 。 


5.3.1 ”while 循环 语句 
while 循环 的 语法 格式 如 下 : 
while (条 件 判断 语句 ) { 
执行 语句 ; 

} 


其 中 当 “ 条 件 判断 语句 ”为 true 时 ， 执 行 后 面 的 
“执行 语句 ”， 然 后 返回 到 条 件 表达 式 继续 进行 判断 ， 直 ， 
到 表达 式 的 值 为 假 ， 才 能 跳出 循环 ， 再 执行 后 面 的 语句 。 < > 结束 
while 循环 语句 的 控制 流程 如 图 5-9 所 示 。 
【 例 5.5】 使 用 while 循环 语句 (示例 文件 ch05\5.5.php)。 I 
程序 块 














<?php 

Snum = 1; 

$str = "15 以 内 的 奇数 为 : "; 

while ($num <=15){ 图 5-9 while 循环 语句 的 控制 流程 


if(Snum % 2!= 0){ 





$str .= Snum-” "7 
Snurm++7 
} 
echo $str; 
> 


二 由 5 当时 








;二 4 二 ~- DD x 
程序 运行 结果 如 图 5-10 所 示 。 @ [ro Bren | 
文件) ” 注 志 5) ” 坦 看 (V) 次 芒 天 [A| 工具] 祁山 (H) 

【案例 训 析 】 15 以 内 的 奇数 为 : 1 3 5 7 9 11 13 15 





本 例 主要 实现 15 以 内 的 奇数 输出 。 从 1~15 依次 判 
断 是 否 为 奇数 ， 如 果 是 ， 则 输出 ; 如 果 不 是 ， 则 继续 下 
一 次 循环 。 WE 


i 图 5-10 使 用 while 循环 语句 
5.3.2 ”do...while 循环 语句 YY 


do...while 循环 的 语法 格式 如 下 : 


dof{ 

执行 语句 ; 

}while (条 件 判 断 语句 ) 

首先 执行 do 后 面 的 “执行 语句 ”， 其 中 的 变量 会 随 着 命令 的 执行 而 发 生变 化 。 当 此 变量 
通过 while 后 的 “条 件 判断 语句 ”判断 为 false 时 ， 将 停止 循环 执行 “执行 语句 ”。 

do...while 循环 语句 的 控制 流程 如 图 5-11 所 示 。 





车 震 址 册 注 夷 

















一 | 程序 人 
1 | 
条 件 
| 
结束 


图 5-11 ”do...while 循环 语句 的 控制 流程 
【 例 5.6】 使 用 do...while 循环 语句 (示例 文件 ch05\5.6.php)。 


<?php 

aa 0 // 声 明 一 个 整数 变量 $aa 

while ($aa != 0){ // 使 用 while 循环 输出 
echo "不 会 被 执行 的 内 容 "; // 这 句 不 会 被 输出 

} 

dof{ // 使 用 do. . .while 循环 输出 
echo "被 执行 的 内 容 "; // 这 句 会 被 输出 
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}while($aa != 0); 

2 

程序 运行 结果 如 图 5-12 所 示 。 从 结果 可 以 看 出 ，while e 
循环 语句 和 do...while 循环 语句 有 很 大 的 区 别 。 ET 





滨 和 (5] 二 看 V) 收 起 (A| 工具 MT 和 和 盈 () 














被 执行 的 内 容 
5.3.3 for 循环 语句 
E25% 
for 循环 的 语法 格式 如 下 : 5-12 ”使 用 do.…while 循环 语句 
for (exprl; expr2; expr3) 
{ 
命令 语句 ; 


其 中 exprl 为 条 件 的 初始 值 ，expr2 为 判断 的 最 终 值 ， 通 常 都 是 用 比较 表达 式 或 逻辑 表达 
式 充当 判断 的 条 件 ， 执 行 完 命令 语句 后 ， 再 执行 expr3。 
for 循环 语句 的 控制 流程 如 图 5-13 所 示 。 


开始 












































图 5-13 for 循环 语句 的 控制 流程 
【 例 5.7】 使 用 for 循环 语句 (示例 文件 ch05\5.7.php)。 


- 0 x 
<?php [OS pocdte. P - ¢| @ iocahon 
for($i=0; $i<4; $i++){ 


文件 (F) 。 兰 强 (E) 查看 (V) 收藏 夫 (A] 工具 MT) 者 动 (H) 
echo "过 思 牵 牛 星 ， 扣 议 河 汉 女 。<br/>"; 迅 表 牛 星 ， 谤 胶 河 汉 女 。 





} 过 表达 牛 星 ， 胶 胶 河 汉 女 。 

2 过 过 这 牛 旺 ， 通 胶 河 汉 女 。 

误 沼 牵 牛 星 ， 接 蚁 河 汉 女 。 

程序 运行 结果 如 图 5-14 所 示 。 从 中 可 以 看 出 ， 语 句 执 i 
行 了 4 次 。 





5-14 ”使 用 for 循环 语句 


5.3.4 foreach 循环 语句 


foreach 语句 是 十 分 常用 的 一 种 循环 语句 ， 它 经 常 被 用 来 遍历 数组 元 素 ， 其 语法 格式 如 下 : 


foreach (数组 as 数组 元 素 ) { 

对 数组 元 素 的 操作 命令 ; 

, 

可 以 把 数组 分 为 两 种 情况 ， 即 不 包含 键 值 的 数组 和 包含 键 值 的 数组 。 
(1) 不 包含 键 值 的 数组 : 


foreach (数组 as 数组 元 素 值 ) { 
对 数组 元 素 的 操作 命令 ; 
. 


(2) 包含 键 值 的 数组 : 


foreach (数组 as 键 值 => 数组 元 素 值 ) { 、 
对 数组 元 素 的 操作 命令 ; NN 
和 

每 进行 一 次 循环 ， 当 前 数组 元 素 的 值 就 会 被 赋值 给 数组 元 素 值 变 量 ， 数 组 指针 会 逐一 地 
移动 ， 直 到 遍历 结束 为 止 。 

【 例 5.8】 使 用 foreach 循环 语句 (示例 文件 ch05\5.8.php)。 


<?php 
$arr = array(" 苹 果 "，" 香 蓄 "， "西红柿 ") ; 


foreach ($arr as $value) 


音 辟 查理 局 测 一 一 可 过 守 冰 下 对 前 二 5 沁 国 








echo "水 果 名 称 : " . $value . "<br />"; 


D> 


程序 运行 结果 如 图 5-15 所 示 。 从 中 可 以 看 出 ， 语 句 执 © > 








行 了 3 次 。 a 
水 朵 名 称 : 各 
5.3.5 ”流程 控制 的 另 一 种 书写 格式 术 全 名称: 并 
在 一 个 含有 多 条 件 、 多 循环 的 语句 中 ， 包含 多 个 {}， 肥 125% ~ 





查看 起 来 比较 烦琐 。 流 程控 制 语句 的 另外 一 种 书写 方式 是 5-15 使 用 foreach 循环 语句 
以 “:” 来 代替 左边 的 大 括号 ， 使 用 endif、endwhile、 
endfor 和 endswitch 来 蔡 代 右边 的 大 括号 ， 这 种 描述 程序 结构 的 可 读 性 比较 强 。 
例如 常见 的 格式 如 下 。 
(1) 于 语句 : 
if (条 件 判断 语句 ) : 
执行 语句 1; 
elseif (条件 判 断 语句 ) : 
执行 语句 2; 
elseif (条 件 判断 语句 ) : 
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执行 语句 3; 

else: 

执行 语句 n7 

endif; 

(2) switch 语句 : 
switch epi : 


case 判断 : a 
执行 语句 1; 
case 判断 结果 Pb: 
执行 语句 2; 





Ge 
执行 语句 n; 


endswitch; 


(3) while 循环 : 
while (条 件 判断 语句 ) : 
执行 语句 ; 


endwhile; 

(4) do...while 循环 : 

do 

命令 执行 语句 ; 

while (条 件 判 断 语句 ) ; 

(5) for 循环 : 

for (初始 化 语句 ; 条 件 终止 语句 ; 增幅 语句 ) : 


执行 语句 

endfor; 

【 例 5.9】 使 用 流程 控制 的 另 一 种 书写 格式 (示例 文件 ch05\5.9.php)。 
<?php 

Smixnum = 1; 

Smaxnum = 10; 


$tmparr[][] = array(); 
$tmparr[0] [0] = 1; 
for($i = 1; $i < $maxnum; Si++) : 
Or ON 全 信人 二 1 
if($j == 0 or $j == $1): 
$tmparr[$i] [$j] = 1; 
else: 
S$tmparr[$i] [$j] = $tmparr[$i - 1] [$j - 1] + $tmparr[$i - 1] [$j]; 
endif; 
endfor; 
endfor; 
foreach ($tmparr as $value): 
foreach ($value as $v1) 
echo RSwL TO 
Cho "<p> 
endforeach; 
Eb 











全 上 
第 
程序 运行 结果 如 图 5-16 所 示 。 从 中 可 以 看 出 ， 该 代码 使 用 新 的 书写 格式 实现 了 杨辉 三 角 
的 排列 输出 。 
= 口 x 和 
[< [BCE Ty | 的 
立体 昌 “将 吉日” 吾 看 ds 工具 m。 才 岗 (H) 执 
人 ~ 行 | 
万 
pt 向 
121 
1331 程 
序 
14641 控 
15101051 a 
1615201561 构 
172135352171 
18285670562881 
19 36 84 126 126 84 369 1 v 
成 125% 一 








图 5-16 ”使 用 流程 控制 的 另 一 种 书写 格式 


5.3.6 ”使 用 break/continue 语句 跳出 循环 


break 关键 字 用 来 跳出 (也 就 是 终止 ) 循 环 控制 语句 和 条 件 控制 语句 中 的 switch 控制 语句 的 
执行 。 例 如 : 





while (++$n) { 
switch ($n) { 
case 1: 
echo "case one"; 
break; 
case 2: 
echo "case two"; 
break 2; 
default: 
echo "case three™s 
break 1; 
} 
下 


人 

在 这 段 程序 中 ，while 循环 控制 语句 里 面包 含 一 个 switch 流程 控制 语句 。 在 程序 执行 到 
break 语句 时 ，break 会 终止 执行 switch 语句 ， 或 者 是 终止 执行 switch 和 while 语句 。 其 中 ， 
case 1 下 的 break 语句 跳出 了 switch 语句 。case 2 下 的 break 2 语句 跳出 switch 语句 和 包含 
switch 的 while 语句 。default 下 的 break 1 语句 与 case 1 下 的 break 语句 一 样 ， 只 是 跳出 switch 
语句 。 这 里 ，break 后 所 携带 的 数字 参数 是 指 break 要 跳出 的 控制 语句 结构 的 层 数 。 

使 用 continue 关键 字 的 作用 是 ， 跳 开 当 前 的 循环 兴 代 项 ， 直 接 进 入 到 下 一 个 循环 迭代 
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项 ， 继 续 执行 程序 。 下 面 通 过 一 个 示例 来 说 明 此 关键 字 的 作用 。 
【 例 5.10】 使 用 continue 关键 字 ( 示 例文 件 ch05\5.10.php)。 
<?php 
$n = 07 
while ($n++ < 6) { 
if ($n == 2){ 
continue; 


下 
echo $n."<br />"; 


} 


2> 
NN 程序 运行 结果 如 图 5-17 所 示 。 | oT rT 


DS 【案例 剖析 】 MD (EV) BA) I 内 0H) 


1 
continue 关键 字 在 n 等 于 2 的 时 候 跳 离 本 次 循环 , 并 
且 直 接 进 入 到 下 一 个 循环 从 代 项 ， 即 当 n 等 于 3。 另 外 ， 6 

continue 关键 字 和 break 关键 字 一 样 ， 都 可 以 在 后 面 直接 跟 LE 
一 个 数字 参数 ， 用 来 表示 跳 开 循 环 的 结构 层 数 。continue 图 5-17 使 用 continue 关键 字 
与 continue 1 相同 。continue 2 表示 跳 离 所 在 循环 和 上 一 级 

循环 的 当前 迭代 项 。 











5.4 ”案例 实战 1 一 一 条 件 分 支 结构 的 应 用 


下 面 的 例子 将 模拟 酒店 管理 系统 中 的 对 人 员 数 目的 判断 。 这 里 使 用 各 种 条 件 分 支 结构 的 
方法 。 
【 例 5.11】 综合 应 用 条 件 分 支 结构 (示例 文件 ch05\5.11.php)。 


<?php 

$members = Null; 

function checkmembers ($members){ 

if ($members < 1){ 

echo "我 们 不 能 为 少 于 一 人 的 顾客 提供 房间 。<br/>"; 

jelsef 

echo "欢迎 来 到 派克 斯 酒店 。<br />"; 

} 

} 

checkmembers (2); 

checkmembers (0.5); 

function checkmembersforroom($members){ 

if ($members < 1){ 

echo A Ae <be /op 
}elseif( $members == 1 

echo "欢迎 来 到 派克 斯 酒店 。 我 们 将 为 您 准备 单 订房 <br /> 
}elseif( $members == 2 ){ 

echo "欢迎 来 到 派克 斯 酒店 。 我 们 将 为 您 准备 标准 间 。<br />"; 


jelseif( $members == 3 ){ 


echo "欢迎 来 到 派克 斯 酒店 。 我 们 将 为 您 准备 三 床 房 。<br />"; 


jelsef 


echo "请 直接 电话 联系 我 们 ， 我 们 将 依照 具体 情况 为 您 准备 合适 的 房间 。<br />"; 


下 

checkmembersforroom(1); 

checkmembersforroom(2); 

checkmembersforroom(3); 

checkmembersforroom(5); 

function switchrooms ($members){ 

switch ($members){ 

case 1: 

echo "欢迎 来 到 派克 斯 酒店 。 我 们 将 为 您 准备 单 床 房 。<br />"; 
break; 

case 2: 

echo "欢迎 来 到 派克 斯 酒店 。 我 们 将 为 您 准备 标准 间 。<br />"; 
break; 

case 3: 

echo "欢迎 来 到 派克 斯 酒店 。 我 们 将 为 您 准备 三 床 房 。<br />"; 
break; 

default: 

echo "请 直接 电话 联系 我 们 ， 我 们 将 依照 具体 情况 为 您 准备 合适 的 房间 。" 
break; 

3 

1 

Switchrooms (1); 

Switchrooms (2) 7 

Switchrooms (3) 7 

Switchrooms (5) 7 

2 


程序 运行 结果 如 图 5-18 所 示 。 








@ [区 -| 人 Smr localhost/code/cho5/: PD ~ S| 七 localhost x 
文件 (F) 坊 强 (E) 查看 (V) ”收藏 交 (A) 工具 (T) 帮助 (H) 
欢迎 来 到 派克 斯 酒店 。 

我 们 不 能 为 少 于 一 人 的 顾客 提供 房间 。 
欢迎 来 到 派克 斯 酒店 。 我 们 将 为 您 准备 单 床 房 。 
欢迎 来 到 派克 斯 酒店 。 我 们 将 为 您 准备 标准 间 。 
欢迎 来 到 派克 斯 酒店 。 我 们 将 为 您 准备 三 床 房 
请 直接 电话 联系 我 们 ， 我 们 将 依照 具体 情况 为 您 准备 合适 的 房间 。 
欢迎 来 到 派克 斯 酒店 。 我 们 将 为 您 准备 单 床 房 。 
欢迎 来 到 派克 斯 酒店 。 
欢迎 来 到 派克 斯 酒店 。 准 名 

请 直接 电话 联系 我 们 ， 我 们 将 优 直 具体 情况 为 算 准 备 合 适 的 房间 。 























有 125% ~ 





图 5-18 综合 应 用 条 件 分 支 结构 
【案例 剖析 】 
其 中 最 后 4 行 由 switch 语句 实现 。 其 他 输出 均 由 站 语 句 实现 。 
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5.5 “案例 实战 2 一 一 循环 控制 结构 的 应 用 


下 面 以 遍历 已 订房 间 门 牌号 为 例 ， 介 绍 循环 控制 语句 的 应 用 技巧 。 
【 例 5.12】 综 合 应 用 循环 控制 结构 (示例 文件 ch05\5.12.php)。 


<?php 

$bookedrooms = array("102", "202";"203")"303", "307"),s 
for ($i=0; $i<5; $i++){ 

echo $bookedrooms[$i]."<br />"; 

} 

function checkbookedroom while ($bookedrooms){ 
$i = 07 

while (isset ($bookedrooms [$i])){ 

echo $i.":".$bookedrooms[$i]."<br />"; 

$i++2 

} 

} 

checkbookedroom while ($bookedrooms); 

$i = 0; 

dof{ 

echo $i."-".$bookedrooms[$i]."<br />"; 

号 i++ 

} while($i < 2); 

对 去 


程序 运行 结果 如 图 5-19 所 示 。 





@ 苇 Wmp/hocahoweodeich DG]| 妆 akee 
HN NS EV A IRM NO 
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5-19 ”综合 应 用 循环 控制 结构 





【案例 剖析 】 


其 中 ，102 到 307 由 for 循环 实现 。0:102 到 4:307 由 while 循环 实现 。0-102 和 1-202 由 
do...while 循环 实现 。for 循环 和 while 循环 都 完全 遍历 了 数组 Sbookedrooms， 而 do...while 循 
环 由 于 while($i < 2)， 所 以 do 后 面 的 命令 执行 了 2 次 。 


5.6 疑难 解 惑 


疑问 1: PHP 中 跳出 循环 的 方法 有 几 种 ? 


答 : PHP 中 的 循环 结构 大 致 有 for 循环 ，while 循环 ，do...while 循环 及 foreach 循环 几 
种 ， 不 管 哪 种 循环 ， 在 PHP 中 跳出 循环 大 致 有 以 下 几 种 方式 。 

1) continue 

continue 用 在 循环 结构 中 ， 控 制程 序 放弃 本 次 循环 continue 语句 之 后 的 代码 并 转 而 进行 
下 一 次 循环 。continue 本 身 并 不 跳出 循环 结构 ， 只 是 放弃 这 一 次 循环 。 如 果 在 非 循环 结构 中 
(如 站 语句 或 switch 语句 中 ) 使 用 continue， 程 序 将 会 出 错 。 

2) break 

break 是 被 用 在 各 种 循环 和 switch 语句 中 的 。 它 的 作用 是 跳出 当前 的 语法 结构 ， 执 行 下 面 
的 语句 。break 语句 可 以 带 一 个 参数 n， 表 示 跳 出 循环 的 层 数 。 如 果 要 跳出 多 重 循环 ， 则 可 以 
用 n 来 表示 跳出 的 层 数 ， 如 果 不 带 参数 ， 则 默认 是 跳出 本 重 循环 。 

3) goto 

goto 实际 上 只 是 一 个 运算 符 ， 和 其 他 语言 一 样 ，PHP 中 也 不 鼓励 滥用 goto， 因 为 滥用 
goto 会 导致 程序 的 可 读 性 严重 下 降 。goto 的 作用 是 将 程序 的 执行 从 当前 位 置 跳 转 到 其 他 任意 
位 置 。goto 本 身 并 没有 要 结束 循环 的 作用 ， 但 其 跳 转 位 置 的 作用 使 得 其 可 以 作为 跳出 循环 使 
用 。 但 PHP 5.3 及 以 上 版 本 停止 了 对 goto 的 支持 ， 所 以 应 该 尽量 避免 使 用 goto。 

4) exit 

exit 是 用 来 结束 程序 执行 的 。 可 以 用 在 任何 地 方 ， 本 身 没 有 跳出 循环 的 含义 。exit 可 以 带 
一 个 参数 ， 如 果 参 数 是 字符 串 ，PHP 将 会 直接 把 字符 串 输出 ， 如 果 参 数 是 integer 整 型 (范围 是 
0 一 254)， 那 么 参数 将 会 被 作为 结束 状态 使 用 。 

5) returm 

retur 语句 是 用 来 结束 一 段 代 码 并 返回 一 个 参数 的 。 可 以 从 一 个 函数 里 调用 ， 也 可 以 从 
一 个 include0) 或 者 require0 语 句 包含 的 文件 里 调用 ， 还 可 以 是 在 主 程序 里 调用 。 如 果 是 从 函数 
里 调用 程序 将 会 马上 结束 运行 并 返回 参数 。 如 果 是 从 include0 或 者 require0 语 句 包含 的 文件 中 
调用 ， 程 序 执行 将 会 马上 返回 到 调用 该 文件 的 程序 ， 而 返回 值 将 作为 include0 或 者 require0 的 
返回 值 。 而 如 果 是 在 主 程序 中 调用 ， 那 么 主 程序 将 会 马上 停止 执行 。 


疑问 2: 循环 体内 使 用 的 变量 ， 定 义 在 哪个 位 置 好 ? 


答 : 在 PHP 语言 中 ， 如 果 变量 要 多 次 使 用 ， 而 且 变 量 的 值 不 改变 ， 建 议 将 该 变量 定义 在 
循环 体 以 外 ; 否则， 就 将 该 变量 定义 在 循环 体内 比较 好 。 
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特殊 的 元 素 集合 
一 一 数组 


数组 在 PHP 中 是 极为 重要 的 数据 类 型 。 本 章 介 绍 什 么 是 数组 、 数 组 包含 的 类 
型 、 数 组 的 构造 ， 以 及 遍历 数组 、 数 组 排序 、 在 数组 中 添加 和 删除 元 素 、 查 询 数 组 
中 的 指定 元 素 、 统 计数 组 元 素 的 个 数 、 删 除数 组 中 重复 的 元 素 、 数 组 的 序列 化 等 操 
作 。 通 过 本 章 的 学 习 ， 读 者 能 够 掌握 数组 的 常用 操作 和 使 用 技巧 。 
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6.1 什么 是 数组 


什么 是 数组 ? 数组 就 是 被 命名 的 用 来 储存 一 系列 数值 的 地 方 。 数 组 array 是 非常 重要 的 数 
据 类 型 。 相 对 于 其 他 数据 类 型 ， 它 更 像 是 一 种 结构 ， 而 这 种 结构 可 以 储存 一 系列 数值 。 

数组 中 的 数值 被 称 为 数组 元 素 。 而 每 一 个 元 素 都 有 一 个 对 应 的 标识 ， 也 称 作 键 值 。 通 过 
这 个 标识 ， 可 以 访问 数组 元 素 。 数 组 的 标识 可 以 是 数字 ， 也 可 以 是 字符 串 。 

例如 ， 一 个 班级 通常 有 十 几 个 人 ， 如 果 需 要 找 出 某 个 学 生 ， 可 以 利用 学 号 来 区 分 每 一 个 学 
生 。 这 时 ， 班 级 就 是 一 个 数组 ， 而 学 号 就 是 下 标 。 如 果 指明 学 号 ， 就 可 以 找到 对 应 的 学 生 。 





6.2 数组 类 型 
数组 分 为 数字 索引 数组 和 关联 索引 数组 。 下 面 详细 讲述 这 两 种 数组 的 使 用 方法 。 


6.2.1 数字 索引 数组 


数字 索引 数组 是 最 常见 的 数组 类 型 ， 默 认 从 0 开始 。 数 组 变量 可 以 随时 创建 和 使 用 。 

声明 数组 的 方法 有 如 下 两 种 。 

(1) 使 用 arrayO 函 数 声明 数组 。 

声明 数组 的 语法 格式 如 下 : 

array 数组 名 称 ( [mixead]) 

其 中 参数 mixed 的 语法 为 key=>value。 如 果 有 多 个 mixed， 可 以 用 逗号 分 开 ， 分 别 定义 了 
索引 和 值 : 

$arr = array ("1"=>" 空 调 "，"2"=>" 冰 箱 "，"3"=>" 洗 衣 机 "，"4"=>" 电 视 机 ") ; 

利用 arrayO 函 数 来 定义 比较 方便 和 灵活 ， 可 以 只 给 出 数组 的 元 素 值 ， 而 不 必 给 出 键 值 。 
例如 : 

$arr = array ("空调 "，" 冰 箱 "， "洗衣机 "，" 电 视 机 ") ; 

(2) 通过 直接 为 数组 元 素 赋值 的 方式 声明 数组 。 

如 果 在 创建 数组 时 不 知道 数组 的 大 小 ， 或 者 数组 的 大 小 可 能 会 根据 实际 情况 发 生变 化 ， 
此 时 可 以 使 用 直接 赋值 的 方式 声明 数组 。 


例如 : 

$arr[1] = “空调 "7 

$arr[2] = "冰箱 " 

$arr[3] = "洗衣 机 " 

$arr[4] = "电视 机 " 

下 面 举 例 进行 讲解 。 

【 例 6.1】 创 建 和 使 用 数组 (示例 文件 ch06\6.1.php)。 
<?php 


$a = array(' 苹 果 ',' 香 燕 ',' 菠 萝 ',' 葡 萄 '); 


echo $a[0]."\t".$a[1]-"\t"-$a[2]-"\t"-$a[3] ."<br/>"7 
echo "$a[0] $a[1] $a[2] $a[3] <br/>"; 

$a[0] = ' 西 瓜 '; 

echo "$a[0] $al[l1l] $a[l2] $a[l3]<br/>"; 

这 


程序 运行 结果 如 图 6-1 所 示 。 
【案例 剖析 】 @ Er -ome 


文 # 虽 。 竺 弓 日 ” 坦 看 M) 交 荣 夫 (A) 工具 帮助 (H) 


(1) 这 里 ，$a 为 一 维 数 组 ， 用 关键 字 array 声 ”苹果 知 茶 菠萝 葡萄 








明 。 并 且 用 “=” 赋 值 给 数组 变量 $a。 二 入 半天 淫乱 
(2) (苹果 香蕉 "' 沙 萝 ,葡萄 ) 为 数组 元 素 ， 且 这 i 
些 元 素 为 字符 串 型 ， 用 单 引号 "方式 表示 。 每 个 数组 元 i 


素 用 “,” 分 开 。echo 命令 直接 打印 数组 元 素 ， 元 素 索 
引 默 认 从 0 开始 ， 所 以 第 一 个 数组 元 素 为 Sa[0]。 
(3) 数组 元 素 可 以 直接 通过 “=” 号 赋值 ， 如 $a[0] = ' 西 瓜 ; ，echo 打印 后 为 “西瓜 ”。 


6.2.2 关联 索引 数组 


关联 索引 数组 的 键 名 可 以 是 数值 和 字符 串 混 合 的 形式 ， 而 不 像 数 字 索 引 数组 的 键 名 只 能 
为 数字 。 所 以 判断 一 个 数组 是 否 为 关联 索引 数组 的 依据 是 : 数组 中 的 键 名 是 否 存在 一 个 不 是 
数字 的 ， 如 果 存 在 ， 则 为 关联 索引 数组 。 

下 面 以 使 用 关联 索引 数组 编写 商品 价格 为 例 进行 讲解 。 

【 例 6.2】 使 用 关联 索引 数组 (示例 文件 ch06\6.2.php)。 

<?php 

$prices = array(' 冰 箱 '=> 3650, ' 洗 衣 机 '=> 2600， 

' 空 调 '=> 2888, ' 电 视 '=> 5888); 
echo "电视 机 的 价格 为 :" .$prices[' 电 视 '] ." 元 "; 








- 0O x 
Ey @ "reocahe.. P ~ | Biocalbost 
程序 运行 结果 如 图 6-2 所 示 。 文件 (有 ”加 缠 (5) 查看 VM) 收藏 天 (A) 工具 人 T) 帮助 (H) 


电视 机 的 价格 为 ，5888 元 
【案例 剖析 】 


这 里 ，echo 命令 直接 指定 数组 $prices 中 的 关键 字 索 引 ' 电 
视 机 (是 个 字符 串 ) 便 可 打印 出 数组 元 素 5888( 是 一 个 整 型 数 )。 


6.3 数组 的 结构 
按照 数组 的 结构 来 划分 ， 可 以 把 数组 分 为 一 维 数组 和 多 维 数组 。 


6.3.1 一 维 数组 


数组 中 每 个 数组 元 素 都 是 单个 变量 ， 不 管 是 数字 索引 还 是 联合 索引 ， 这 样 的 数组 为 一 维 
数组 。 


R125% ~ 


6-2 ”使 用 关联 索引 数组 
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【 例 6.3】 创 建 一 维 数组 (示例 文件 ch06\6.3.php)。 


<?php 
$a = array(' 冰 箱 ',' 洗 衣 机 ',' 空调 ', ' 电视 '); 


$prices = array(' ' 冰 箱 ' => 3650," 洗衣 机 ， => 2600, ' 空 调 '=> 2888, ' 电 视 '=> 5888); 
2 


其 中 的 $a 和 Sprices 都 是 一 维 数组 。 
6.3.2 多维 数 组 


数组 也 是 可 以 嵌 套 的 ， 即 每 个 数组 元 素 也 可 以 是 一 个 数组 ， 这 种 含有 数组 的 数组 就 是 多 
维 数组 。 例 如 : 


<?php 
$roomtypes = array (array ('type'=>' 单 床 房 '， 
'info'=>' 此 房间 为 单 人 单间 。'， 
'price per day'=>298 
), 
array ('type'=>' 标 准 间 '， 
"info'=>' 此 房间 为 两 床 标准 配置 。'， 
'price per day'=>268 
NW 
array ('type'=>' 三 床 房 '， 
1 info'=>' 此 房间 备 有 三 张 床 '， 
'price per day'=>198 
I 
array ('type'=>'VIP 套房 '， 
'info'=>' 此 房间 为 VIP 两 间 内 外 套房 ' ， 
'price per day'=>368 
) 





); 
和 
其 中 的 Sroomtypes 就 是 多 维 数组 。 这 个 多 维 数组 其 实 包 含 了 两 个 维 数 。 有 点 像 数 据 库 的 
表格 ， 在 第 一 个 array 里 面 的 每 个 数组 元 素 都 是 一 个 数组 ， 而 这 些 数组 就 像 是 数据 二 维 表 中 的 
一 行 记录 。 这 些 包 含 在 第 一 个 array 里 面 的 array 又 都 包含 3 个 数组 元 素 ， 分 别 是 3 个 类 型 的 
信息 ， 这 就 像 是 数据 二 维 表 中 的 字段 。 
上 面 的 数组 如 果 绘 制 成 图 ， 效 果 如 图 6-3 所 示 。 











| ) A B C D 
type info Price per_day 

单 床 房 此 房间 为 单 人 单间 。 298 array 

标准 间 此 房间 为 两 床 标准 配置 。 268 array 


三 床 房 此 房间 备 有 三 张 床 198 array 
司 司 368 
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6-3 ”二 维 数组 的 直观 图 示 


其 实 ，$roomtypes 就 代表 了 这 样 一 个 数据 表 。 
也 可 能 出 现 两 维 以 上 的 数组 ， 比 如 三 维 数组 。 例 如 : 


<?php 


U 


$building = array (array (array('type'=>" 单 床 房 '， 

"info'=>' 此 房间 为 单 人 单间 。， 
"price per day'=>298 
Ji 

array ('type'=>' 标 准 间 '， 
'info'=>' 此 房间 为 两 床 标准 配置 。' 
"price per day'=>268 
Ys 

array ('type'=>' 三 床 房 '， 
"info'=>' 此 房间 备 有 三 张 床 '， 
'price per day'=>198 
), 

array ('type'=>'VIP 套房 '， 
"info'=>' 此 房间 为 VIP 两 间 内 外 套房 '， 
'price per day'=>368 
» 
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ys 
array (array ('type'=>' 普 通 餐 厅 包 房 '， 
'info'=>' 此 房间 为 普通 餐厅 包 房 。' 
"Foomid'=>201 
)， 
array ('type'=>' 多 人 餐厅 包 房 '， 
"'info'=>' 此 房间 为 多 人 和 餐厅 包 房 。'， 
'roomid'=>206 
)， 
array('type'=>! 豪 华 餐 厅 包 房 '， 
"'info'=>' 此 房间 为 豪华 餐厅 包 房 。' 
"roomid'=>208 
), 
array ('type'=>'VIP 和 餐厅 包 房 '， 
'info'=>' 此 房间 为 VIP 餐厅 包 房 '， By 
"roomid'=>310 


) 


AG 





A ); 

这 个 三 维 数组 在 原来 的 二 维 数组 后 面 又 增加 了 一 个 二 维 数组 ， 给 出 了 餐厅 包 房 的 数据 二 
维 表 信息 。 把 这 两 个 二 维 数组 作为 更 外 围 array 的 两 个 数组 元 素 ， 就 产生 了 第 三 维 。 这 个 表述 
等 于 用 两 个 二 维 信息 表 表 示 了 一 个 名 为 Sbuilding 的 数组 对 象 ， 如 图 6-4 所 示 。 





A 有 C D E 
type price_per_day 










单 床 房 直 刘 对 时 入 单间 . 298 array 
标准 间 此 房间 为 两 床 标准 配置 。 268 array 
三 床 房 array 





二 张 床 
此 六 二 内 外 套 六 


ee array 
此 启 间 eT 208 array 











ARRAY (三 维 ) 





6-4 ”三 维 数组 的 直观 图 示 
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64 遍历 数组 
所 谓 数组 的 遍历 ， 是 要 把 数组 中 的 变量 什 读 取出 来 。 下 面 讲述 常见 的 遍历 数组 的 方法 。 


6.4.1 遍历 一 维 数字 索引 数组 


下 面 讲 解 如 何 通过 循环 语句 遍历 一 维 数字 索引 数组 。 此 案例 中 使 用 到 了 for 循环 ， 以 及 
foreach 循环 。 
【 例 6.4】 遍 历 一 维 数字 索引 数组 (示例 文件 ch06\6.4.php)。 


<?php 

$a = array(' 草 果 '，,' 香 燕 ', ' 西 瓜 ', ' 葡 萄 ') ; 
for ($i=0; $i<3; $i++){ 

echo $a[$i] ."” (for 循环 ) <br />"; 
foreach ($a as $b){ 

echo $b."(foreach 循环 ) <br/>"; 


人 


程序 运行 结果 如 图 6-5 所 示 。 





= 
-She/he 5 - © [Glccahor 

| SH WED EV WN IRM Wb) 

苹果 (for 循环 ) 

香 烙 (for 循 环 ) 

西瓜 (for 循环 ) 

苹果 (foreach 循 环 ) 








香 茧 (foreach 循 环 ) 
西瓜 (foreach 循 环 ) 
短 萄 (foreach 循 环 ) 


用 125% ~ 


6-5 ”遍历 一 维 数字 索引 数组 





【案例 剖析 】 


(1) for 循环 只 进行 了 0、1、2,， 共 3 次 。 
(2) foreach 循环 则 列 出 了 数组 中 所 有 的 数组 元 素 。 


6.4.2 ”遍历 一 维 联合 索引 数组 


下 面 以 遍历 商品 价格 为 例 ， 对 联合 索引 数组 进行 遍历 。 

【 例 6.5】 遍 历 一 维 联合 索引 数组 (示例 文件 ch06\6.5.php)。 

<?php 

$prices = array(' 冰 箱 '=> 3650, ' 洗 衣 机 '=> 2600, ' 空 调 '=> 2888, ' 电 视 '=> 5888); 


foreach ($prices _as $ps){ 
echo S$Sps-" 元 "."<br/>"7 


站 
foreach ($prices as S$key => $value){ 


echo $key.":".$value." 元 "." 每 个 。 <br />"; 


reset ($prices); 
while ($element = each($prices)){ 
echo $element['key']."\t"; 
echo $element['value'] ." 元 "; 
echo "<br/>"; 
} 
reset ($prices_); 
while (list($type, $ps) = each($prices)){ 
echo "$type - $ps"." 元 "."<br/>"; 
} 


区 3 


程序 运行 结果 如 图 6-6 所 示 。 
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5888 元 

冰箱 :3650 元 每 个 
洗衣 机 :2600 元 每 个 。 
空调 :2888 元 每 个 。 
电视 :5888 元 每 个 . 
冰箱 3650 元 


空调 2888 元 
电视 5888 元 
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图 6-6 ”遍历 一 维 联合 索引 数组 


【案例 剖析 】 


(1) foreach ($prices as $ps){} 遍历 了 数组 元 素 ， 所 以 输出 4 个 整 型 数值 。 而 foreach 


($prices as $key => $value) 联 则 除了 遍历 数组 元 素 外 ， 还 遍历 了 其 所 对 应 的 关键 字 ， 如 ' 冰 箱 ' 是 
数组 元 素 3650 的 关键 字 。 


(2) 这 段 程序 中 使 用 了 while 循环 ， 还 用 到 了 几 个 新 的 函数 reset0、eachO0 和 list0。 由 于 


【 例 6.6】 遍 历 多 维 数 组 (示例 文件 ch06\6.6.php)。 


在 前 面 的 代码 中 ，S$prices 已 经 被 foreach 循环 遍历 过 ， 而 内 存 中 的 实时 元 素 为 数组 的 最 后 一 个 
元 素 ， 因 此 ， 如 果 想 用 while 循环 来 遍历 数组 ， 就 必须 用 reset0 函 数 ， 把 实时 元 素 重新 定义 为 
数组 的 开头 元 素 。eachO0 则 是 用 来 遍历 数组 元 素 及 其 关键 字 的 函数 。list0 是 把 each0 中 的 值 分 
开 赋 值 和 输出 的 函数 。 


6.4.3 遍历 多 维 数组 
下 面 以 使 用 多 维 数组 编写 酒店 房价 类 型 为 例 进行 遍历 。 
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<2php 

Sroomtypes = array (array('type'=>' 单 床 房 '， 
'info'=>' 此 房间 为 单 人 单间 。' 
'price per day'=>298 


), 
array('type'=>" 标准 间 '， 
"info'=>" 此 房间 为 两 床 标准 配置 。 
'price per day'=>268 
), 
array('type'=>" 三 床 房 '， 
"info '=>' 此 房间 备 有 三 张 床 '， 
'price per day'=>198 





array( ‘type'=>'VIP 套房 '， 
"info'=>' 此 房间 为 VIP 两 间 内 外 套房 '， 
'price per day'=>368 


) 
?7 
for ($row=0; $row<4; $row++){ 
while (list($key, $value) = each($roomtypes[$row])){ 
echo "$key:$value"."\t |"; 
echo '<br />'; 
D> 


程序 运行 结果 如 图 6-7 所 示 。 


@ ep /ochonfeod noo 6 D ~ © | caon 
底 怕 月。 妨 强 [F) 下 看 (V) 必需 关 A) 工具 (大多 [H) 


间 为 单 人 单 问 、 





type; 单 床 房 |info:j 






price per_day:298 

3 |price_per_day:268 

Y nfo: 此 房间 备 有 三 张 床 |price_per_day:198 

type:T 和 info: 二 套房 |price_per_day:368 








图 6-7 遍历 多 维 数组 
【案例 剖析 】 
(1) Sroomtypes 中 的 每 个 数组 元 素 都 是 一 个 数组 ， 而 作为 数组 元 素 的 数组 又 都 有 3 个 拥 
有 键 名 的 数组 元 素 。 


(2) 使 用 for 循环 配合 eachO 、listO 函 数 来 遍历 数组 元 素 ， 便 得 到 输出 。 


6.5 数组 排序 
下 面 主要 讲述 如 何 对 一 维 数组 和 多 维 数 组 进行 排序 操作 。 


6.5.1 一 维 数组 排序 


下 面 通 过 示例 展示 如 何 对 数组 进行 排序 。 
【 例 6.7】 一 维 数组 排序 (示例 文件 ch06\6.7.php)。 


<?php 
$roomtypes = array(' 单 床 房 ', ' 标 准 间 ',' 三 床 房 ', 'VIP 套房 ' ) ; 
$prices per day = 
array (' 单 床 房 '=> 298, ' 标 准 间 '=> 268, ' 三 床 房 '=> 198, 'VIP 套房 '=> 368); 
Sort ($roomtypes); 
foreach ($roomtypes as S$key => $value){ 
echo $key.":".$value."<br />"; 
asort ($prices per day); 
foreach ($prices per day as $key => $value){ 
echo $key.":".$value." 每 日 。<br />"; 
3 
ksort ($prices_per day); 
foreach ($prices per day as $key => $value){ 
echo $key.":".$value." 每 天 。<br />"; 
' 
rsort ($roomtypes); 
foreach ($roomtypes as S$key => $value){ 
echo $key.":".$value."<br />"; 
下 
arsort ($prices per_day); 
foreach ($prices per day as $key => $value){ 
echo $key.":".$value." 每 日 。<br />"; 
+ 
krsort ($prices per_day); 
foreach ($prices per day as $key => $value){ 
echo $key.":".$value." 每 天 。<br />"; 


一 


党 : 


程序 运行 结果 如 图 6-8 所 示 。 








@ 局 hapVWecahesycode/cweE7Php ~ 6 | 居 iocehost x 
六 作坊 和 (E)】 下 看 V) 收 夫 入 工具 帮 动 | 








林 房 

房 :198 每 日 . 
标准 问 :268 每 日 
单 床 房 :298 每 日 





单 床 房 :298 每 天 。 
三 床 房 :198 每 天 . 







2: 标 
3:VIP 套 房 

VIP 套房 :368 每 日 。 

单 床 房 :298 每 日 - 

标准 则 :268 每 日 

三 床 房 :198 每 日 。 

房 :198 每 天 。 

单 床 房 :298 每 天 。 

标准 同 :268 每 天 。 

VIP 套房 :368 每 天 . v 


二 125% 












6-8 ”对 一 维 数组 进行 排序 
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【案例 剖析 】 


(1) 这 段 代码 是 关于 数组 排序 的 内 容 ， 涉 及 sort0 、asortO0 、ksortO 、rsortO0 、arsortO 、 
krsort()。 其 中 ，sort0 是 默认 排序 。asort0 根 据 数 组 元 素 的 值 的 升序 排序 。ksort0 是 根据 数组 元 
素 的 键 值 ， 也 就 是 关键 字 的 升序 排序 。 

(2) rsortO0、arsortO0、krsort0 则 正好 与 所 对 应 的 升序 排序 相反 ， 都 为 降序 排序 。 


6.5.2 ”多维 数组 排序 


对 于 一 维 数组 ， 通 过 sort0 等 一 系列 排序 函数 就 可 以 对 它 进行 排序 。 而 对 于 多 维 数组 ， 排 
序 就 没有 那么 简单 了 。 首 先 需要 设 定 一 个 排序 方法 ， 也 就 是 建立 一 个 排序 函数 ， 再 通过 usortO 
函数 对 特定 数组 采用 特定 排序 方法 进行 排序 。 下 面 通过 案例 介绍 多 维 数组 排序 。 

【 例 6.8】 多 维 数组 排序 (示例 文件 ch06\6.8.php)。 

<?php 

$roomtypes = array (array('type'=>' 单 床 房 '， 


"info'=>" 此 房间 为 单 人 单间 。， 
'price per_day'=>298 





), 
array('type'=>"' 标准 间 '， 
"info'=>" 此 房间 为 两 床 标准 配置 。， 
"price_per_day'=>268 
)， 
array ('type'=>' 三 床 房 '， 
'info'=>' 此 房间 备 有 三 张 床 ' ， 
'price per_day'=>198 
)， 
array('type'=>'VIP 套房 '， 
'info'=>' 此 房间 为 VIP 两 间 内 外 套房 ' ， 
'price per_day'=>368 
) 
); 
function compare ($x, $y){ 
if ($x['price per day'] == $y['price per day']){ 
return 0; 
J}else if ($x['price per day'] < $y['price per day']){ 
TEUrD =13 
}elsef{ 
return 1; 
: 
| 


Usort ($roomtypes, 'compare') 7 


for ($row=0; $row<4; Srow++) { 

reset ($roomtypes [$row]); 

while (list($key, $value) = each($roomtypes[$row])){ 
echo "$key:$value"."\t |"; 


echo "<br />'; 
?> 


Se 






程序 运行 结果 如 图 6-9 所 示 。 


| | 贺 合 mex//ocalhosveode /dos D - © | 人 cealhot 

奖 件 (F] 护 汤 5] 前 吾 V) 校训 去 A) 工具 (等 蔓 (HI 

type: 三 床 房 |info: 此 房间 备 有 三 张 床 |price_per_day:198 
type: 标 准 同 |info; 此 房 同 ee price_ per_day:268 
type: 单 床 房 |info: 此 房 同 为 单 ce_per_day:298 
type:VIP 套 房 |info: 兹 生生 机 外 村 price_per_day:368 








3% 
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图 6-9 对 多 维 数组 进行 排序 
【案例 剖析 】 
(1) 函数 compare0 定 义 了 排序 方法 ， 通 过 对 price_per_day 这 一 数组 元 素 的 对 比 进行 排 
序 。 然 后 usort0 采 用 compare 方法 对 $roomtypes 这 一 多 维 数组 进行 排序 。 
(2) 如 果 这 个 排序 的 结果 是 正 向 排序 ， 怎 么 进行 反 向 排序 呢 ? 这 就 需要 对 排序 方法 进行 
调整 。 其 中 ，recompare0 就 是 上 一 段 程序 中 compare0 的 相反 判断 ， 同 样 采用 usortO 函 数 输出 
后 ， 得 到 的 排序 正好 与 前 一 段 程序 输出 顺序 相反 。 


6.6 字符 串 与 数组 的 转换 


使 用 explode 和 implode 函数 来 实现 字符 串 和 数组 之 间 的 转换 。explode 用 于 把 字符 串 按 
一 定 的 规则 拆 分 为 数组 中 的 元 素 ， 并 且 形 成 数组 。implode 函数 用 于 把 数组 中 的 元 素 按 照 一 
定 的 连接 方式 转换 为 字符 串 。 
【 例 6.9】 使 用 explode 和 implode 函数 来 实现 字符 串 和 数组 之 间 的 转换 (示例 文件 
ch06\6.9.php)。 
<?php 
$prices_per day = 
array (' 单 床 房 '=> 298, ' 标 准 间 '=> 268, ' 三 床 房 '=> 198, 'VIP 套房 '=> 368); 
echo implode(' 元 每 天 / '，, $prices_per day).'<br />'; 


$roomtypes =' 单 床 房 ,标准 间 , 三 床 房 , VIP 套房 '; 


print_r(explode(',',$roomtypes)); 
















> 
程序 运行 结果 如 图 6-10 所 示 。 
@ @- Gooehoweode/h "ey 
298 元 每 天 / 368 
Array ( os A 3 标准 间 [2] => 三 床 房 [3] => VIP 套 房 ) 
Rl 
6-10 使 用 explode 和 implode 函数 来 实现 字符 串 和 数组 之 间 的 转换 
【案例 训 析 】 


(1) Sprices_per_day 为 数组 。implode(' 元 每 天 /，$prices_per_day) 对 $prices_per_day 中 的 数 
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组 元 素 中 间 添 加 连接 内 容 ， 也 叫 元 素 胶水 (gluej)， 把 它们 连接 成 一 个 字符 串 输出 。 这 个 元 素 胶 
水 只 在 元 素 之 间 。 

(2) $roomtypes 为 一 个 由 “,” 号 分 开 的 字符 串 。explode(,,，$roomtypes) 确 认 分 隔 符 为 
“,” 号 后 ， 以 “,” 号 为 标记 把 字符 串 中 的 字符 分 为 4 个 数组 元 素 ， 并 且 生 成 数组 返回 。 


6.7 ”向 数组 中 添加 和 删除 元 素 





数组 创建 完成 后 ， 用 户 还 可 以 继续 添加 和 删除 元 素 ， 从 而 满足 实际 工作 的 需要 。 


NN 6.7.1 向 数组 中 添加 元 素 


数组 是 数组 元 素 的 集合 。 如 果 向 数组 中 添加 元 素 ， 就 像 是 往 一 个 盒子 里 面 放 东 西 。 这 就 
涉及 “先进 先 出 ”或 是 “后 进 先 出 ”的 问题 。 

(1) 先进 先 出 有 点 像 排队 买 火车 票 。 先 进 到 购买 窗口 区 域 的 ， 购 买 完成 之 后 从 旁边 的 出 口 
出 去 。 

(2) 后 进 先 出 有 点 像 是 给 枪 的 弹 夹 上 子弹 。 最 后 装 上 的 那 一 颗 子 弹 是 要 最 先 打出 去 的 。 

PHP 对 数组 添加 元 素 的 处 理 使 用 push、pop、shift 和 unshift 函数 来 实现 ， 可 以 实现 先进 
先 出 ， 也 可 以 实现 后 进 先 出 。 

下 面 通 过 例子 介绍 在 数组 前 面 添加 元 素 ， 以 实现 后 进 先 出 。 

【 例 6.10】 在 数组 前 面 添加 元 素 以 实现 后 进 先 出 (示例 文件 ch06\6.10.php)。 

<?php 

$clients = array (' 李 丽 丽 ', ' 赵 大 勇 ',' 方 芳 芳 ' ) ; 

array_unshift ($clients,，' 王 小 明 ', ' 刘 小 帅 '); 


print_r($clients); 
信之 


程序 运行 结果 如 图 6-11 所 示 。 











抒 [=] x 
[@"| reo/nocahoscodetdhos/s 10php Po) localhost 沪 @ 
HN PE) EEV)| daly IRM Wi 
Array 〈 [0] => 王小明 [1] =》 刘 小 帅 [2] = 李 丽 丽 [3] => 赵 大 勇 [4] =》 方芳 芳 ) | 
Riz 





图 6-11 实现 后 进 先 出 
【案例 剖析 】 
(1) 数组 $clients 原本 拥有 3 个 元 素 。array_unshift0) 向 数组 $clients 的 头 部 添加 了 数组 元 素 
' 王 小 明 '、' 刘 小 帅 '。 最 后 通过 print_r0 输 出 ， 通 过 其 数字 索引 可 以 知道 添加 元 素 的 位 置 。 
(2) array_unshift0 函 数 的 语法 格式 如 下 : 
array_unshift (目标 数组 ， [和 欲 添加 数组 元 素 1， 欲 添加 数组 元 素 2，. . .]) 


用 同样 的 例子 介绍 在 数组 后 面 添 加 元 素 ， 以 实现 先进 先 出 。 


【 例 6.11】 在 数组 后 面 添加 元 素 以 实现 先进 先 出 (示例 文件 ch06\6.11.php)。 
<?php 
$clients = array(' 李 丽 丽 ', ' 赵 大 勇 ',' 方 芳 芳 ' ) ; 
array_push ($clients，' 王 小 明 ',' 刘 小 帅 '); 
print r($clients); 
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汪汪， 
程序 运行 结果 如 图 6-12 所 示 。 
一 口 x 
D7 0 | 二 localhodt x| 
Array ( [0] =》 李 丽 丽 [1] => 起 大 勇 [2] =》 方 芳 芳 [3] => 王小明 [4] => 刘 小 帅 ) 
ET 
图 6-12 实现 先进 先 出 
【 娄 例 剖析 】 


(1) 数组 $clients 原本 拥有 3 个 元 素 。array_push0 向 数组 Sclients 的 尾部 添加 了 数组 元 素 ' 
王小明 '、' 刘 小 帅 '。 最 后 通过 print_r0 输 出 ， 通 过 其 数字 索引 可 以 知道 添加 元 素 的 位 置 。 

(2) array_push0) 函 数 的 语法 格式 如 下 : 

array_push (目标 数组 ， [ 欲 添加 数组 元 素 1， 欲 添加 数组 元 素 2，.. .]) 

push 就 是 “ 推 ” 的 意思 ， 这 个 过 程 就 像 是 排队 的 时 候 把 人 从 队伍 后 面向 前 推 。 


6.7.2 ”从 数组 中 删除 元 素 


从 数组 中 删除 元 素 是 添加 元 素 的 逆 过 程 。 PHP 使 用 array_shift0 和 array_popO 函 数 分 别 从 


数组 的 头 部 和 尾部 删除 元 素 。 
下 面 通过 例子 介绍 如 何在 数组 前 面 删除 第 一 个 元 素 并 返回 该 元 素 值 。 
【 例 6.12】 在 数组 前 面 删除 第 一 个 元 素 并 返回 该 元 素 值 (示例 文件 ch06\6.12.php)。 
<?php 
$services = array (' 洗 衣 ', ' 订 和 餐 ',' 导 游 ',' 翻 译 '); 
$deletedservices = array_shift($services); 
echo $deletedservices."<br />"; 
print r($services); 








程序 运行 结果 如 图 6-13 所 示 。 





© Breerones po Broates 
文件 F)】 编 转 (E) 得 郑 (V) ”收藏 六 。 工具 (T) 基色 (H) 





洗衣 
Array 〈 [0] => 订餐 [1] => 导游 [2] => 翻译 ) 


图 6-13 ”在 数组 前 面 删除 第 一 个 元 素 并 返回 该 元 素 值 
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【 娄 例 剖析 】 了 
(1) 数组 $services 原本 拥有 4 个 元 素 。array_shift0 从 数组 Sservices 的 头 部 删除 了 第 一 个 
数组 元 素 ， 并 且 直 接 把 所 删除 的 元 素 值 返回 ， 且 赋值 给 了 变量 $deletedservices。 最 后 通过 echo 
输出 $deletedservices， 并 用 print_r0 输 出 $services。 
(2) array_shiftO 函 数 仅仅 删除 目标 数组 的 头 一 个 数组 元 素 。 它 的 语法 格式 如 下 : 
array_shift (目标 数组 ) 


以 上 例子 为 数字 索引 数组 ， 如 果 是 带 键 值 的 联合 索引 数组 ， 它 的 效果 相同 ， 返 回 所 删除 
元 素 的 元 素 值 。 

下 面 用 同样 的 例子 介绍 如 何在 数组 后 面 删除 最 后 一 个 元 素 并 返回 该 元 素 值 。 

【 例 6.13】 在 数组 后 面 删除 最 后 一 个 元 素 并 返回 该 元 素 值 (示例 文件 ch06\6.13.php)。 

<?php 

$services = array('sl'=>' 洗 衣 ', 's2'=>' 订 和 餐 ','s3'=>' 导 游 ', 's4'=>' 翻 译 '); 

$deletedservices = array pop($services); 

echo $deletedservices."<br/>"; 

print _r($services); 

2 


程序 运行 结果 如 图 6-14 所 示 。 








@ 车 .hapyiocaihosyco PD ~ © | Blocalbost 
文件 (F) 六 名 (E) 各 看 (V) 收 基 夫 (A) 工具 (D) 相助 (H) 











翻译 
Array ( [sl] =>》 洗衣 [s2] =》 订 餐 [s3] =》 导游 ) 
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图 6-14 在 数组 后 面 删除 最 后 一 个 元 素 并 返回 该 元 素 值 
【案例 剖析 】 

(1) 数组 $services 原本 拥有 4 个 元 素 。array_ popO 从 数组 $services 的 尾部 删除 了 最 后 一 个 
数组 元 素 ， 并 且 直 接 把 所 删除 的 元 素 值 返回 ， 且 赋值 给 了 变量 $deletedservices。 最 后 通过 echo 
输出 $deletedservices， 并 用 print_r0 输 出 $services。 

(2) array_pop0 函 数 仅仅 删除 目标 数组 的 最 后 一 个 数组 元 素 。 它 的 语法 格式 如 下 : 

array_pop (目标 数组 ) 


这 个 例子 中 的 数组 是 一 个 联合 索引 数组 。 


6.8 查询 数组 中 的 指定 元 素 


数组 是 一 个 数据 集合 。 能 够 在 不 同类 型 的 数组 和 不 同 结构 的 数组 内 确定 某 个 特定 元 素 是 
否 存在 ， 是 必要 的 。PHP 提供 in_array0、array key_exists0 、array_search0 、array keys0 和 
array_values() 函 数 ， 可 以 按照 不 同 的 方式 来 查询 数组 元 素 。 


下 面 通过 例子 介绍 如 何 查询 数字 索引 数组 和 联合 索引 数组 ， 并 且 都 是 一 维 数组 。 
【 例 6.14】 查 询 数字 索引 数组 和 联合 索引 数组 (示例 文件 ch06\6.14.php)。 


<?php 

$roomtypes = array(' 单 床 房 ',' 标 准 间 '，,' 三 床 房 ', 'VIP 套房 ') ; 
$prices per day = 

array (' 单 床 房 '=> 298, ' 标 准 间 '=> 268, ' 三 床 房 '=> 198, 'VIP 套房 '=> 368); 
if (in_array (' 单 床 房 ', $roomtypes)){ 

echo ' 单 床 房 元 素 在 数组 $sroomtypes 中 。<br />'; 

if (array_key_exists(' 单 床 房 ', $prices_per day)){ 

echo ' 键 名 为 单 床 房 的 元 素 在 数组 $prices_per_day 中 。<br />'; 

} 

if(array_search (268, $prices_ per day)){ 

echo ' 值 为 268 的 元 素 在 数组 Sprices_per_day 中。<br />'; 
S$prices per day_keys = array_keys($prices per day); 
print_r($prices per day_ keys); 

$prices per day values = array values ($prices per day); 
print_r($prices per day values); 

2 


程序 运行 结果 如 图 6-15 所 示 。 


站 攻关 汕 芭 六 才 协 9 游 关 





七 htpy/iocahosVcode/chosl514php 万 -6 要 bcahes 
文件 (志和 看 WV) 收 基 (A) 工具 帮助 H) 





‘oomtypes 中 。 

组 $prices_per_day 中 。 

Ev ee day 中 - 

Array ( [0] =》 单 床 房 [1] => 标准 间 [2] =》 三 床 房 [3] => VIP 套 房 ) 
Array ( [0] => 298 1] => 268 [2] => 198 [3] => 368 ) 
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图 6-15 查询 数字 索引 数组 和 联合 索引 数组 
【案例 剖析 】 


(1) 数组 Sroomtypes 为 一 个 数字 索引 数组 。in_array(' 单 床 房 ,$Sroomtypes) 判 定 元 素 ' 单 床 
房 ' 是 否 在 数组 Sroomtypes 中 ， 如 果 在 ， 则 返回 true。 站 语句 返回 值 为 真 ， 便 打印 结果 。 

(2) 数组 Sprices_per_day 为 一 个 联合 索引 数组 。array_key_exists(' 单 床 房 ',$prices_per_day) 
判定 一 个 键 值 为 单 床 房 ' 的 元 素 是 否 在 数组 Sprices_per_day 中， 如果 在 ， 则 返回 tue。 让 语句 得 
到 返回 值 为 真 ， 便 打印 结果 。array_key_exists0 是 专门 针对 联合 数组 的 键 名 进行 查询 的 函数 。 

(3) array_search0 是 专门 针对 联合 数组 的 元 素 值 进行 查询 的 函数 。 同 样 ， 针 对 数组 
Sprices_per_day 这 个 联合 数组 ，array_search(268.$prices_per_day) 判 定 一 个 元 素 值 为 268 的 元 
素 是 否 在 数组 $prices per day 中 ， 如 果 在 ， 则 返回 trme。 半 语句 返回 值 为 真 ， 便 打印 结果 。 

(4) 函数 array_keys0 取 得 数组 的 键 值 ， 并 把 键 值 作为 数组 元 素 输出 为 一 个 数字 索引 数 
组 ， 主 要 用 于 联合 索引 数组 。array_keys($prices_per_day) 获 得 数组 Sprices_per_day 的 键 值 ， 并 
把 它 赋 值 给 变量 $prices_per_day_keys， 用 print r() 打 印 结果 。 函 数 array_keys0 虽 然 也 可 以 取 
得 数字 索引 数组 的 数字 索引 ， 但 是 这 样 意义 不 大 。 
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(5) 函数 array_values0 取 得 数组 元 素 的 元 素 值 ， 并 把 元 素 值 作为 数组 元 素 输出 为 一 个 数 
字 索 引 数 组 。array_values($prices_per_day) 获 得 数组 Sprices_per_day 的 元 素 值 ， 并 把 它 赋 值 给 
变量 $prices_ per _day_values， 用 print r0 打 印 结果 。 

这 几 个 函数 只 是 针对 一 维 数组 ， 无 法 用 于 多 维 数组 ， 它 们 在 查询 多 维 数组 的 时 候 ， 会 只 
处 理 最 外 围 的 数组 ， 其 他 内 嵌 的 数组 都 作为 数组 元 素 处 理 ， 不 会 得 到 内 说 数组 内 的 键 值 和 元 
素 值 。 





6.9 统计 数组 元 素 的 个 数 


下 面 通过 例子 介绍 如 何 用 countO 函 数 来 统计 数组 元 素 的 个 数 。 
【 例 6.15】 使 用 count0 函 数 统计 数组 元 素 的 个 数 (示例 文件 ch06\6.15.php)。 


<?php 
$prices per day = 
array (' 单 床 房 '=> 298, ' 标 准 间 '=> 268, ' 三 床 房 '=> 198, 'VIP 套房 '=> 368); 
$roomtypesinfo = array (array('type'=>' 单 床 房 ' ， 
"'info'=>' 此 房间 为 单 人 单间 。'， 
'price per_day'=>298 
), 
array ('type'=>' 标 准 间 '， 
"'info'=>' 此 房间 为 两 床 标准 配置 。'， 
"Price_per_day'=>268 
)， 
array ('type'=>' 三 床 房 '， 
"'info'=>' 此 房间 备 有 三 张 床 '， 
'price per_ day'=>198 
i 
array ('type'=>'VIP 套房 '， 
"info'=>' 此 房间 为 VIP 两 间 内 外 套房 '， 
'price per day'=>368 
) 
hb 
echo count ($prices_per_day) . ' 个 元 素 在 数组 $prices_per_day 中 。<br />'; 
echo count ($roomtypesinfo) . ' 个 内 尾数 组 在 二 维 数组 Sroomtypesinfo 中 。<br />'; 
echo count ($roomtypesinfo,1) . ' 个 元 素 在 Sroomtypesinfo 中 。<br />'; 
区 


程序 运行 结果 如 图 6-16 所 示 。 


| oo x 
@ 敬 htpi/localhostcode P - 忆 | 疗 localhost | 
文件 (F) ”篇 辑 (E) ”查看 (V】 收音 夫 [A) 工具 (T) 邦 却 (9) 

4 个 元 素 在 数组 $prices_per_day 中 。 

4 个 内 嵌 数 组 在 二 维 数组 Sroomtypesinfo 中 。 
16 个 元 素 在 Sroomtypesinfo 中 。 
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6-16 用 count() 函 数 来 统计 数组 元 素 的 个 数 





【 娄 倒 剖析】 
(1) 数组 $Sprices_ per day 通过 count0 函 数 返回 整数 4。 因为 数组 $prices_per_ day 有 4 个 数 
组 元 素 。 
(2) 数组 $roomtypesinfo 是 二 维 数 组 。count($Sroomtypesinfo) 只 统计 了 数组 Sroomtypesinfo 
内 的 4 个 内 婴 数 组 的 数量 。 
(3) echo count($roomtypesinfo,1) 语 句 中 ，count0) 函数 设置 了 一 个 模式 (mod) 为 整数 
“1”。 这 个 模式 (mod) 设 置 为 整数 “1” 的 意义 是 ，count 统计 的 时 候 要 对 数组 内 部 所 有 的 内 
嵌 数 组 进行 循环 查询 。 所 以 最 终 的 结果 是 所 有 内 嵌 数 组 的 个 数 加 上 内 赃 数 组 内 元 素 的 个 数 ， 
是 4 个 内 嵌 数 组 加 上 12 个 数组 元 素 ， 为 16。 
使 用 array_count valuesO 函 数 对 数组 内 的 元 素 值 进行 统计 ， 并 且 返 回 一 个 以 函数 值 为 键 
值 、 以 函数 值 个 数 为 元 素 值 的 数组 。 
【 例 6.16 】 使 用 array_count values0 函 数 来 统计 数组 的 元 素 值 个 数 (示例 文件 ch06\ 
6.16.php)。 


<?php 

$prices_per_day = array(' 单 床 房 '=> 298, ' 标 准 间 '=> 268, ' 三 床 房 '=> 198， 
"四 床 房 '=> 198, 'VIP 套房 '=> 368); 

print_r(array_count values ($prices per _day)); 

2 


程序 运行 结果 如 图 6-17 所 示 。 


旦 由 路 关 沙 汪 王 六 二 二 9 直 国 





A 








- 0O x 
@ httpi/localhost/cede P - © | 詹 locahost 
文件 朋 。 妨 涡 (日 查看 VV) 收藏 夫 (A】 工具 (Tm 帮助 ff) 
Array ( [298] => 1 [268] => 1 [198] =>2 [368] => 1 ) 
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图 6-17 使 用 array_count_values() 函 数 来 统计 数组 的 元 素 值 个 数 


【案例 剖析 】 
(1) 数组 Sprices_per_day 为 一 个 联合 数组 ， 通 过 array_count_values($prices_per_day) 统 计 
数组 内 元 素 值 的 个 数 和 分 布 ， 然 后 以 ( 键 值 和 值 ) 的 形式 返回 出 一 个 数组 。 元 素 值 为 198 的 元 素 


有 两 个 ， 虽 然 它们 的 键 值 完全 不 同 。 
(2) array_count_valuesO 只 能 用 于 一 维 数组 ， 因 为 它 不 能 把 内 嵌 的 数组 当 作 元 素来 统计 。 


6.10 ”删除 数组 中 重复 的 元 素 


使 用 array_unique0 函 数 可 实现 数组 中 元 素 的 唯一 性 ， 也 就 是 去 掉 数 组 中 重复 的 元 素 。 不 
管 是 数字 索引 数组 还 是 联合 索引 数组 ， 都 是 以 元 素 值 为 准 。array_unique0 函 数 返回 具有 唯一 


性 元 素 值 的 数组 。 
【 例 6.17】 用 array_unique0 函 数 去 掉 数 组 中 重复 的 元 素 (示例 文件 ch06\6.17.php)。 


<?php 
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S$prices_per_dqay = array (' 单 床 房 '=> 298, ' 标 准 间 '=> 268, ' 三 床 房 '=> UIDR 
' 四 床 房 '=> 198, 'VIP 套房 '=> 368); 

$prices_per_day2 = array(' 单 床 房 '=> 298, ' 标 准 间 '=> 268, ' 四 床 房 '=> 198， 
' 三 床 房 '=> 198, 'VIP 套房 '=> 368); 

print rl(array unique ($prices per day)); 

print rl(array unique($prices per day2)); 

了 全 


程序 运行 结果 如 图 6-18 所 示 。 


轩 -| 夸 ntpyllocalhostcode/cho6/517php DP 0) localhost x | 
文件 (F)” 编 铝 (E) 章 看 (V)】 收藏 夫 (A) 工具 (T) 帮 有 纯 (H) 


Array ( [ 单 床 房 ] =》298 [标准 间 ] => 268 [三 床 房 ] =》198 [VIP 套房 ] =》368 ) 
Array ( [ 单 床 房 ] => 298 [标准 间 ] => 268 [四 床 房 ] => 198 [VIP 套房 ] =》368 ) 
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6-18 用 array_unique() 函 数 去 掉 数 组 中 重复 的 元 素 
【案例 剖析 】 


数组 Sprices_per_day 为 一 个 联合 索引 数组 ， 通 过 array_unique($prices_per_day) 去 除 重复 的 
元 素 值 。array_unique0 函 数 去 除 重复 的 值 是 去 除 第 二 个 出 现 的 相同 值 。 由 于 $prices_per_day 
与 $prices_per_day2 数组 中 ， 键 值 为 “三 床 房 ”和 键 值 为 “四 床 房 ”的 198 元 素 的 位 置 正好 相 
反 ， 所 以 对 两 次 输出 的 所 保留 的 值 也 正好 相反 。 


6.11 调换 数组 中 的 键 值 和 元 素 值 


可 以 使 用 array_flip0 函 数 调换 数组 中 的 键 值 和 元 素 值 。 

【 例 6.18】 用 array_flip0 函 数 调 换 数 组 中 的 键 值 和 元 素 值 (示例 文件 ch06\6.18.php)。 
<?php 

$prices_per_day = array(' 单 床 房 '=> 298, ' 标 准 间 '=> 268, ' 三 床 房 '=> 198， 
"四 床 房 '=> 198, 'VIP 套房 '=> 368); 

print_r(array_flip($prices per_ day)); 

人 


程序 运行 结果 如 图 6-19 所 示 。 





@ [@-|@ mpiocahosycode/chos/s 1aphp 万 - © || localhost 
文件 (| 纺 委 ( 吉 看 (V) 收藏 夫 (A】 工具 (T) 帮助 H) 
Array ( [298] => 单 床 房 [268] => 标准 间 [198] => 四 床 房 [368] => VIP 套 房 ) 
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6-19 用 array_fip() 函 数 调换 数组 中 的 键 值 和 元 素 值 
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【 案 侈 剖析 】 


数组 $prices_per_day 为 一 个 联合 索引 数组 ， 通 过 array_flip($prices_per_day) 调 换 联合 索引 
数组 的 键 值 和 元 素 值 ， 并 且 进 行 返 回 。 但 有 意思 的 是 ，S$prices_per_day 是 一 个 拥有 重复 元 素 
值 的 数组 ， 且 这 两 个 重复 元 素 值 的 键 名 是 不 同 的 。array_flip0 逐 个 调换 每 个 数组 元 素 的 键 值 和 
元 素 值 。 而 如 果 原 来 的 元 素 值 变 为 键 名 以 后 ， 就 有 两 个 原先 为 键 名 的 、 现 在 调换 为 元 素 值 的 
数值 与 之 对 应 。 调 换 后 ，array_flip0 等 于 对 原来 的 元 素 值 (现在 的 键 名 ) 赋 值 。 当 array_flip0 再 
次 调换 到 原来 相同 的 、 现 在 为 键 名 的 值 时 ， 相 当 于 对 同一 个 键 名 再 次 赋值 ， 则 头 一 个 调换 时 
的 赋值 将 会 被 覆盖 ， 显 示 的 是 第 二 次 的 赋值 。 


上 相关 小 二 世 站 二 址 9 兴国 


6.12 ”数组 的 序列 化 


数组 的 序列 化 (Serialize) 是 用 来 将 数组 的 数据 转换 为 字符 串 ， 以 便于 传递 和 进行 数据 库存 
储 。 而 与 之 相对 应 的 操作 就 是 反 序列 化 (Unserialize)， 把 字符 串 数据 转换 为 数组 加 以 使 用 。 

【 例 6.19】 使 用 serializeO 函 数 和 unserialize0 函 数 (示例 文件 ch06\6.19.php)。 

<?ph] 

$arr = array ("王小明 '，' 李 本 珊 '， 方芳 芳 '， 刘 小 帅 '，' 张 大 勇 '，' 张 明明 ) ; 

$str = serialize ($arr); 

echo $str."<br /><br />"; 

$new_arr = unserialize ($str); 

print_r($new arr); 

人 


程序 运行 结果 如 图 6-20 所 示 。 








口 x | 
@ http://localhost/code/ch06/6.19.php D7 Clocalhost *| 2 
文件 (篇 恕 (E) 豆 看 VV) 收藏 赤 (A) 工具 (1) 考 辐 (H) 
a:6: 全 :0;s:6:“ 王 小 明 ”;i:1;s:6:” 李 丽 丽 ”;i:2;s:6: “方芳 芳 ;i:3; s:6:“ 刘 小 帅 ”;i:4;s:6: “张大 
勇 ”;i:5;s:6:“ 张 明明 ”;} 


Array ( [0] = 王小明 [1] = 李 丽 丽 [2] = 方芳 芳 [3] => 刘 小 籼 [4] => 张大 勇 [5] => 
张 明 明 ) 
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图 6-20 使 用 serialize() 函 数 和 unserialize() 函 数 
【案例 剖析 】 


serialize0 和 unserialize0 这 两 个 函数 的 使 用 是 比较 简单 的 ， 通 过 这 样 的 方法 对 数组 数据 进 
行 储存 和 传递 将 会 十 分 方便 。 例 如 ， 可 以 直接 把 序列 化 之 后 的 数组 数据 存放 在 数据 库 的 某 个 
字段 中 ， 在 使 用 时 再 通过 反 序 列 化 进行 处 理 。 
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6.13 疑难 解 惑 


疑问 1: 数组 的 合并 与 联合 有 何 区 别 ? 

答 : 对 数组 的 合并 使 用 array_merge() 函 数 。 两 个 数组 的 元 素 会 合并 为 一 个 数组 的 元 素 。 
而 数组 的 联合 ， 是 指 两 个 一 维 数组 ， 一 个 作为 关键 字 ， 另 一 个 作为 数组 元 素 值 ， 联 合成 为 一 
个 新 的 联合 索引 数组 。 

疑问 2: 如 何 快速 清空 数组 ? 

答 : 在 PHP 中 ， 快 速 清空 数组 的 方法 如 下 : 


arr = array() / /理解 为 重新 给 变量 赋 一 个 空 的 数组 
unset ($arr) // 这 才 是 真正 意义 上 的 释放 ， 将 资源 完全 释放 
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7.1 字符 串 的 单 引 号 和 双 引 号 


字符 串 是 指 一 连 串 不 中 断 的 字符 。 这 里 的 字符 主要 包括 以 下 几 种 类 型 。 

(1) 字母 类 型 。 例 如 常见 的 a、b、c 等 。 

(2) 数字 类 型 。 例 如 常见 的 1、2、3、4 等 。 

(3) 特殊 字符 类 型 。 例 如 常见 的 #、%、^、$ 等 。 

(4) 不 可 见 字 符 类 型 。 例 如 回 车 符 、Tab 字符 、 换 行 符 等 。 

通常 使 用 单 引号 或 双 引 号 来 标识 字符 串 ， 表 面 看 起 来 没有 什么 区 别 。 但 是 ， 对 存在 于 字 


符 串 中 的 变量 来 说 ， 


二 者 是 不 一 样 的 : 双 引 号 内 会 输出 变量 的 值 ， 而 单 引 号 内 则 直接 显示 变 


量 名 称 。 双 引号 中 可 以 通过 “\” 转 义 符 输出 的 特殊 字符 如 表 7-1 所 示 。 


\foctal #} 
\x {hexadecimal #} 


表 7-1 双 引 号 中 可 以 通过 “\” 转 义 符 输 出 的 特殊 字符 


含义 
换行 且 回 到 下 一 行 的 最 前 端 
Tab 
反 和 斜 杠 
ASCII 码 的 0 
把 此 符号 转 义 为 单纯 的 美元 符号 ， 而 不 再 作为 声明 变量 的 标识 符 
换行 
八进制 转 义 
十 六 进 制 转 义 





而 单 引号 中 可 以 通过 “\” 转 义 符 输出 的 特殊 字符 只 有 如 表 7-2 所 示 的 两 个 。 


表 7-2 单 引 号 中 可 以 通过 “\” 转 义 符 输出 的 特殊 字符 


含 义 
转 义 为 单 引号 本 身 ， 而 不 作为 字符 串 标识 符 
反 斜 杠 转 义 为 其 本 身 





下 面 通过 示例 来 讲解 它们 的 不 同 用 法 。 
【 例 7.1】 使 用 双 引 号 和 单 引号 (示例 文件 ch07\7.1.php)。 


<?php 


$message = "PHP 程序 "; 

echo "这 是 关于 字符 串 的 程序 。<br/>"; 

echo “这 是 一 个 关于 双 引 号 和 和 N\$ 的 mesaage<br/>™s 
串 的 程序 。 


Smessage2 = " 


echo " 娩 吕 一个 关于 字符 时 的 程序 <bE/> 
echo ' 这 是 一 个 关于 单 引号 的 $message2'; 


echo $message2; 


Ye 


程序 运行 结果 如 图 7-1 所 示 。 可 见 单 引号 和 双 引 号 在 PHP 中 处 理 普 通 的 字符 串 时 效果 是 
一 样 的 ， 而 在 处 理 变量 时 是 不 一 样 的 。 单 引号 中 的 内 容 只 是 被 当成 普通 的 字符 串 处 理 ， 而 双 
引号 中 的 内 容 是 可 以 被 解释 并 蔡 换 的 。 





ist 扎 x 有 
区 -Spoke D+ oCB oclhor 

文件 (站 ”篇 澡 (E) 查看 V) 收藏 夫 /A) 工具 () 孝 同 (H) 

这 是 关于 字符 串 的 程序 。 

这 是 一 个 关于 双 引 号 和 $ 的 PHP 程 序 

这 是 一 个 关于 字符 串 的 程序 。 

这 是 一 个 关于 单 引 号 的 $message2 字 符 弟 的 程序 








或 125% > 





届 专 必 一 一 满 涝 计 首 否 演 习习 习 出 人 小 鲁 


图 7-1 单 引号 和 双 引 号 的 区 别 
【案例 剖析】 
(1) 第 一 段 程序 使 用 双 引 号 对 字符 串 进行 处 理 。\$ 转 义 成 了 美元 符号 。$message 的 值 
"PHP 程序 "被 输出 。 
(2) 第 二 段 程序 使 用 单 引 号 对 字符 串 进行 处 理 。$message2 的 值 在 单 引 号 的 字符 串 中 无 法 
被 输出 ， 但 是 可 以 通过 变量 打印 出 来 。 


7.2 ”字符 串 的 连接 符 


字符 串 连接 符 的 使 用 十 分 频繁 。 这 个 连接 符 就 是 “.”( 点 )。 它 可 以 直接 连接 两 个 字符 
串 ， 可 以 连接 两 个 字符 串 变 量 ， 也 可 以 连接 字符 串 和 字符 串 变 量 。 
【 例 7.2】 使 用 字符 串 的 连接 符 (示例 文件 ch07\7.2.php)。 


<?php 

// 定 义 字符 串 

$a = " 桃 查 依 稀 香 暗 渡 。"， 

$b = " 谁 在 秋千 ， 笑 里 轻 轻 语 。"; 

// 连 接 上 面 两 个 字符 串 ， 中 间 用 逗号 分 隔 
$c = $a.$b; ”// 输 出 连接 后 的 字符 串 
echo $c; 

汉学 


程序 运行 结果 如 图 7-2 所 示 。 
除了 上 面 的 方法 以 外 ， 读 者 还 可 以 使 用 {0 方法 连接 字符 串 ， 此 方法 类 似 于 C 语言 中 printf 
的 占 位 符 。 下 面 举 例 说 明 其 使 用 方法 。 
【 例 7.3】 使 用 人 0 方 法 连接 字符 串 ( 示 例文 件 ch07\7.3.php)。 


<?php 

// 定 义 需要 插入 的 字符 帅 

$a 三 "百花 "; 

$b = " 雪 埋 藏 "; 

// 生 成 新 的 字符 串 

$c = "一 朵 忽 先 变 ，{ $a} 丝 后 香 。 欲 传 春 信息 ， 不 怕 {$b}。"; ”// 输 出 连接 后 的 字符 串 
echo $c; 

?> 
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程序 运行 结果 如 图 7-3 所 示 。 
0 x ox 
(< [Som IE re [ES e/eete- P- ©| Brocton > 国 “ 








文人 口交 昌 日” 寺 丰 M_ 收 天 天 AA) 工具 T) NIH) 
桃 奉 依稀 香 暗 渡 。 谁 在 秋千 ， 笑 里 轻 轻 语 。 


文 # 介 ”将 号 日。 查 肆 W)。 收 二 夫人) 工具 mT 才 钝 (H] 











一 条 忽 先 变 ， 百 花 皆 后 香 。 欲 传 春 信息 ， 不 怕 雪 埋藏 


R125% ~ 
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图 7-2 使 用 字符 串 的 连接 符 图 7-3 使 用 {} 方 法 连接 字符 串 


7.3 字符 串 的 基本 操作 


字符 串 的 基本 操作 主要 包括 对 字符 串 的 格式 化 处 理 、 连 接 / 切 分 字符 串 、 比 较 字 符 串 、 字 
符 串 子 串 的 对 比 与 处 理 等 。 


7.3.1 手动 和 自动 转 义 字符 串 中 的 字符 


手动 转 义 字符 串 数据 ， 就 是 在 引号 内 (包括 单 引号 和 双 引 号 ) 通 过 使 用 反 斜 本 “\” 使 一 些 
特殊 字符 转 义 为 普通 字符 。 这 个 方法 在 介绍 单 引 号 和 双 引 号 的 时 候 已 经 有 了 详细 的 描述 。 

自动 转 义 字符 串 的 字符 ， 是 通过 PHP 的 内 置 函数 addslashes0 来 完成 的 。 该 函数 经 常用 于 
格式 化 字符 串 以 实现 MySQL 的 数据 库 储 存 。 

addslashes() 函 数 返回 在 预定 义 字符 之 前 添加 反 斜 杠 的 字符 串 。 预 定义 字符 包括 单 引号 
()、 双 引号 ")、 反 和 斜 杠 Q) 和 NULL。 

【 例 7.4】 使 用 addslashes0O) 函 数 (示例 文件 ch07\7.4.php)。 

<?php 

$str = " 数 点 ' 雨 声 风 约 住 。 腾 腕 淡 月 云 来 去 。"; 


echo $str."<br/>"; 
echo addslashes ($str); 




















2 
程序 运行 结果 如 图 7-4 所 示 。 
一 口 x 
[@-|@ np//ocalho.. P ~ © || @ localhost 
文件 (站 妨 纺 (E) 查看 (V) 收 训 闪 (A)】 工具 (T) 帮助 (H) 
数 点 ' 雨 声 风 约 住 。 腾 脆 淡 月 云 来 去 。 
数 点 \' 雨 声 风 约 住 。 腾 肛 淡 月 云 来 去 。 
村 125% > 








图 7-4 使 用 addslashes() 函 数 
7.3.2 ”计算 字符 串 的 长 度 
计算 字符 串 的 长 度 在 很 多 应 用 中 都 经 常 出 现 ， 比 如 统计 输入 框 输入 文字 的 多 少 等 。 这 个 
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功能 使 用 strlen0 函 数 就 可 以 实现 。 下 面 通过 实例 介绍 计算 字符 串 长 度 的 方法 和 技巧 。 
【 例 7.5】 使 用 stlen0) 函 数 (示例 文件 ch07\7.5.php)。 


<?php 
$aa = " 千 磨 万 击 还 坚 劲 ， 任 尔 东 西南 北 风 。good"; 
$length = strlen($aa); 
if(strlen($aa)>40){ 
echo "输入 的 字符 串 的 长 度 不 能 大 于 100 个 字符 。"; 
}elsef 
echo "此 字符 串 长 度 为 : $length"; 
} 


?> 
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程序 运行 结果 如 图 7-5 所 示 。 Ode 
文件 (月 ”六 岛 (E) 可 看 V) 要 凉 夫 ( 内 工具 (帮助 [HH) 
【 娄 例 剖析 ] 此 字符 串 长 度 为 : 36 
(1) $aa 为 一 个 字符 串 变 量 。strlen($aa) 则 是 直接 调用 
strlen() 函 数 计算 出 字符 串 的 长 度 。 CUT 
(2) 在 站 语句 中 ，strlen($aa) 返 回 字符 串 长 度 并 与 100 


这 一 上 限 做 比较 。 图 7-5 使 用 strlen() 函 数 


(3) $aa 中 有 中 文 和 英文 两 种 字符 。 由 于 每 个 中 文字 符 占 两 个 字符 位 ， 而 每 个 英文 字符 只 
占 一 个 字符 位 ， 且 字符 串 内 的 每 个 空格 也 算 一 个 字符 位 ， 所 以 ， 最 后 字符 串 的 长 度 为 36 个 字符 。 


7.3.3 字符 串 单词 统计 


有 时 候 ， 对 字符 串 的 单词 进行 统计 有 更 大 的 意义 。 使 用 str_word_countO 函 数 可 以 实现 此 
操作 ， 但 该 函数 只 对 基于 ASCII 码 的 英文 单词 起 作用 ， 并 不 对 UTF8 的 中 文字 符 起 作用 。 
下 面 通过 示例 介绍 字符 串 单 词 统计 中 的 应 用 和 技巧 。 
【 例 7.6】 使 用 str_word_countO 函 数 (示例 文件 ch07\7.6.php)。 
<?php 


$aa = "How many words in this sentence? Just count it."; 

$bb = "爆竹 声 中 一 岁 除 ， 春 风 送 暖 入 导 苏 。 千 门 万 户 暗 瞳 日 ， 总 把 新 桃 换 旧 符 。"; 
echo "变量 aa 的 长 度 为 : " .str_word_count ($aa) ."<br/>"; 

echo "变量 bb 的 长 度 为 : " .str_word_count ($bb); 

这 














程序 运行 结果 如 图 7-6 所 示 。 可 见 str_word_count0 函 = 
数 无 法 计算 中 文字 符 ， 查 询 结果 为 0。 人 SEE asclscw 
文件 (F) 。 妨 句 (E) 查看 (V) 收藏 夫 (A) 工具 (T) 帮助 (H) 

7.3.4 ”清理 字符 串 中 的 空格 最 这 的 长 度 为: 9 
空格 在 很 多 情况 下 是 不 必要 的 。 因 此 ， 清 除 字符 串 i 





中 的 空格 显得 十 分 重要 。 例 如 ， 在 判定 输入 是 否 正确 的 
程序 中 ， 出 现 了 不 必要 的 空格 ， 将 增 大 程序 出 现 错误 判 
断 的 概率 。 


7-6 使 用 str_word_count() 函 数 


和 
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清除 空格 要 使 用 到 ltrimn0、rtrimn0 和 trim0 函 数 。 

其 中 ，ltrim0 是 从 左面 清除 字符 串 头 部 的 空格 。rtrim0 是 从 右面 清除 字符 串 尾 部 的 空格 。 
trim() 则 是 从 字符 串 两 边 同时 去 除 头 部 和 尾部 的 空格 。 

下 面 通 过 示例 介绍 去 除 字符 串 中 空格 的 方法 和 技巧 。 

【 例 7.7】 清 理 字符 串 中 的 空格 (示例 文件 ch07\7.7.php)。 

<?php 

$aa = " 朱雀 桥 边 野草 花 ， 乌 衣 埠 口 夕阳 斜 。 "; 

echo "清理 空格 后 :" .1trim($aa) ." 结 尾 <br/>"; 

echo "清理 空格 后 :" .rtrim($aa) ." 结 尾 <br/>"; 

echo "清理 空格 后 :" .trim($aa) ." 结 尾 <zbr/>"; 


$bb = " 旧时 王 谢 堂前 燕 ， 飞 入 寻常 百姓 家 。 "; 
echo "清理 空格 后 :" .trim ($bb) . "结尾 "; 














a 
程序 运行 结果 如 图 7-7 所 示 。 
- OO x 
全 htpy/ocahosticode/ PD ~ © || 居 localhost 
文件 (站 坊 和 6) 坦 看 (V) 收 项 夫 A) 工具 T) 和 有 (H) 
清理 空格 后 : 朱 洗 桥 边 野 草花 ， 乌 衣 巷 口 夕阳 斜 。 结尾 
清理 空格 后 : 朱 佟 桥 边 野 草花 ， 乌 衣 巷 口 夕阳 斜 。 结 尾 
清理 空格 后 : 朱 乱 桥 边 野 草花 ， 乌 衣 巷 口 夕阳 斜 。 结 尾 
清理 空格 后 :旧时 王 谢 堂前 燕 ， 飞 入 寻 常 百 姓 家 。 结 尾 
叶 125% ~ 
图 7-7 清理 字符 串 中 的 空格 
【案例 剖析 】 


(1) $aa 为 一 个 两 端 都 有 空格 的 字符 串 变量 。ltrim($aa) 从 左边 去 除 空格 ，rtrim($aa) 从 右边 
去 除 空格 ， 而 trim($aa) 则 从 两 边 同时 去 除 ， 得 到 这 些 输 出 结果 。 

(2) Sbb 为 一 个 两 端 都 有 空格 且 中 间 也 有 空格 的 字符 串 。 用 trim($bb) 处 理 ， 只 是 去 除了 
两 边 的 空格 。 


7.3.5 ”字符 串 的 切 分 与 组 合 


字符 串 的 切 分 使 用 explode0 和 strtok0 函 数 。 切 分 的 反 向 操作 为 组 合 ， 使 用 implode0 和 
join0 函 数 。 

其 中 ，explode0 把 字符 串 切 分 成 不 同 部 分 后 ， 存 入 一 个 数组 。implodeO 函 数 则 是 把 数组 
中 的 元 素 按照 一 定 的 间隔 标准 组 合成 一 个 字符 串 。 

下 面 通过 示例 介绍 字符 串 切 分 和 组 合 的 方法 和 技巧 。 

【 例 7.8】 字 符 串 的 切 分 与 组 合 (示例 文件 ch07\7.8.php)。 

<?php 

$aa = "How_to_split this sentance"; 

$bb = "向 晚 意 不 适 ， 驱 车 登 古 原 。 夕 阳 无 限 好 ， 只 是 近 黄昏 。"; / /把 这 个 句子 按 空格 拆 分 。 

$a = explode('_',$aa); 

print r($a); 

echo "<br/>"; 


$b = explode(' ',$bb); 

print er(SB)y 

echo "<br/>"; 

echo implode('>',$a) ."<br/>"; // 按 > 号 把 变量 a 分 开 
echo implode('*"',$b); // 按 * 号 把 变量 a 分开 


?> 


程序 运行 结果 如 图 7-8 所 示 。 
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7-8 字符 串 的 切 分 与 组 合 


【案例 剖析 】 

(1) explode0 函 数 把 $aa 和 S$bb 按照 下 划 线 和 空格 的 位 置 分 别 切 分 成 Sa 和 $b 两 个 数组 。 

(2) implode0 函 数 把 Sa 和 $b 两 个 数组 的 元 素 分 别 按照 “>” 为 间隔 和 “*” 为 间隔 组 合成 
新 的 字符 串 。 


7.3.6 字符 串 子 串 的 截取 


在 一 串 字符 串 中 截取 一 个 子 串 ， 就 是 字符 串 截 取 。 

完成 这 个 操作 需要 用 到 substr0 函 数 。 这 个 函数 有 3 个 参数 ， 分 别 规定 了 目标 字符 串 、 起 
始 位 置 和 截取 长 度 。 它 的 语法 格式 如 下 : 

substr (目标 字符 串 ， 起 始 位 置 ， 截 取 长 度 ) 


其 中 ， 目 标 字 符 串 是 某 个 字符 串 变 量 的 变量 名 ， 起 始 位 置 和 截取 长 度 都 是 整数 。 

如 果 都 是 正 数 ， 起 始 位 置 的 整数 必须 小 于 截取 长 度 的 整数 ， 否 则 函数 返回 值 为 假 。 

如 果 截 取 长 度 为 负数 ， 则 意味 着 是 从 起 始 位 置 开始 往 后 除去 从 目标 字符 串 结尾 算 起 的 长 
度数 的 字符 以 外 的 所 有 字符 。 

下 面 通过 示例 介绍 字符 串 截取 的 方法 和 技巧 。 

【 例 7.9】 使 用 substr0 函 数 (示例 文件 ch07\7.9.php)。 


<?php 

$aa = " To the world you may be one Person ™; 
$bb = "杨柳 青青 江水 平 ， 闻 郎 江上 踏歌 声 。"; 

echo substr($aa,0,10)."<br/>"; 

echo substr($aa,1,8)."<br/>"; 

echo substr ($aa,0,-2)."<br/>"; 

seho aubatr ($bbr 0 10) <RE 人 > 

echo substr ($bb,0,8)."<br/>"; 

echo substr($bb,0,11); 
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程序 运行 结果 如 图 7-9 所 示 。 和 
[案例 剖析 】 OE 
(GD Sea 为 英文 字符 串 变 量 。substr($aa,0,10)、 x 局 a you may be one perso 
substr($aa,1,8) 展示 了 起 始 位 置 和 截取 长 度 。 饭 育 并 I 
substr($aa,0,-2) 则 是 从 字符 串 开 头 算 起 ， 除 了 最 后 两 个 。 Ba 
字符 ， 其 他 字符 都 截取 的 子 字符 串 。 
(2) $bb 为 中 文字 符 串 变量 。 因 为 中 文字 符 都 是 图 7-9 使 用 substr() 函 数 


全 角 字 符 ， 占 两 个 字符 位 ， 所 以 截取 长 度 一 定 要 是 偶 
数 ， 如 果 是 奇数 ， 则 在 此 字符 位 上 的 汉字 将 不 被 输出 。 


7.3.7 字符 串 子 串 的 替换 


在 某 个 字符 串 中 蔡 换 其 中 的 某 个 部 分 是 重要 的 应 用 ， 就 像 在 使 用 文本 编辑 器 中 的 替换 功 
能 一 样 。 

完成 这 个 操作 需要 使 用 substr_replace0 函 数 。 它 的 语法 格式 如 下 : 

substr_replace (目标 字符 串 ， 蔡 换 字符 串 ， 起 始 位 置 ， 蔡 换 长 度 ) 


下 面 举 例 介 绍 字 符 串 蔡 换 的 方法 和 技巧 。 
【 例 7.10】 使 用 substr_replaceO 函 数 (示例 文件 ch07\7.10.php)。 


<?php 

$aa = "ID:128565843388654"; 

echo substr_replace ($aay 四 妇女 雪 方 雪 雪 方太 二 三 广 二 my 37 开工 ) ."<br/>"; 
echo substr_replace ($aa, " 尾 号 为 ", 3,11); 

人 





程序 运行 结果 如 图 7-10 所 示 。 Oo EE 
【案例 剖析 】 ee 工 贞 (T) 大 即 (H) 
ID: 尾 号 为 8654 
(1) $aa 字符 串 变量 从 第 3 个 字符 开始 为 ID 号 。 
第 一 个 输出 是 以 “****+s*sests” 蔡 换 第 三 个 字符 开 
始 往 后 的 11 个 字符 。 CT 
本 I 图 7-10 使 用 substr_replace() 函 数 


7.3.8 字符 串 查找 


在 一 个 字符 串 中 查找 另外 一 个 字符 串 ， 就 像 文 本 编辑 器 中 的 查找 功能 一 样 。 实 现 这 个 操 
作 需 要 使 用 strstr0 或 stristr0 函 数 。strstr0 函 数 的 语法 格式 如 下 : 
strstr (目标 字符 串 ， 需 查找 的 字符 串 ) 


当 函 数 找到 需要 查找 的 字符 或 字符 串 时 ， 则 返回 从 第 一 个 查找 到 字符 串 的 位 置 往 后 所 有 


的 字符 串 内 容 。 
stristr0 函 数 为 不 敏感 查找 ， 也 就 是 对 字符 的 大 小 写 不 敏感 。 用 法 与 strstr0 相 同 。 
下 面 通 过 示例 介绍 字符 串 查 找 的 方法 和 技巧 。 
【 例 7.11】 字 符 串 查找 (示例 文件 ch07\7.11.php)。 


<?php 

$aa = "I have a Dream that to find a string with a dream."; 
$bb = "杨柳 青青 江水 平 ， 闻 郎 江 上 踏歌 声 。 东 边 日 出 西边 雨 ， 道 是 无 晴 却 有 上 晴 。"; 
echo strstr($aa,"dream")."<br/>"; 

echo stristr($aa,"dream")."<br/>"; 

echo strstr($aa,"that")."<br/>"; 

echo strstr ($bb, "歌声 ") ."<br/>"; 


程序 运行 结果 如 图 7-11 所 示 。 

一 口 x 
-weiccaho-. P - ©|| 名 loclhost 
给 铝 (E) 查看 (V) 权 苦 夫 (A) 工具 (T) 帮助 (H) 








drean. 

Dream that to find a string with a dreanm. 
that to find a string with a dream. 

歌声 .东边 日 出 西边 两， 道 是 无 哺 却 有 晴 - 


质 125% 
图 7-11 字符 串 查找 
【 业 例 剖析 】 
(1) $aa 为 英文 字符 串 变 量 。strstr($aa，"dream' 对 大 小 写 敏 感 ， 所 以 输出 字符 串 中 最 后 的 


字符 。stristr($aa, "dream") 对 大 小 写 不 敏感 ， 所 以 直接 在 第 一 个 大 写 的 匹配 字符 处 开始 输出 。 
(2) S$bb 为 中 文字 符 串 变量 。strstr0 函 数 同样 对 中 文字 符 起 作用 。 


7.4 疑难 解 惑 


疑问 1: 如何 格 式 化 字符 囊 ? 


答 : 在 PHP 中 ， 有 多 种 方法 可 以 格式 化 字符 串 ， 其 中 用 于 数字 字符 串 格式 化 的 number_ 
format0 函 数 比较 常用 。 该 函数 的 语法 格式 如 下 : 


number_format (number, decimals,decimalpoint, separator) 


该 函数 可 以 有 1 个 、2 个 或 者 4 个 参数 ， 但 不 能 有 3 个 参数 。 各 个 参数 的 含义 如 下 。 

(1) number: 必需 参数 。 要 格式 化 的 数字 。 如 果 没 有 设置 其 他 参数 ， 则 数字 会 被 格式 化 为 
不 带 小 数 点 且 以 逗号 作为 分 隔 符 。 

(2) decimals: 可 选 参数 。 规 定 多 少 个 小 数 。 如 果 设 置 了 该 参数 ， 则 使 用 点 号 作为 小 数 点 
来 格式 化 数字 。 

(3) decimalpoint: 可 选 参数 。 规 定 用 作 小 数 点 的 字符 串 。 
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(4) separator: 可 选 参数 。 规 定 用 作 千 位 分 隔 符 的 字符 串 。 仅 使 用 该 参数 的 第 一 个 字符 。 
例如 以 下 代码 : 


<?php 

echo number format ("1000000"); 

echo "<br/>"; 

echo number format ("1000000",2); 

echo "<br/>"; 

echo number format ("1000000"™,2,",","."); 





程序 运行 结果 如 图 7-12 所 示 。 


= a Xx 
-| me//iocadhe-. P - © | Bocalhost 


文件 月 。 韦 沪 (E) 至 看 NV) 。 收藏 闪 (A) 工具 (TT) 大助 IH) 
1, 000, 000 

1, 000, 000. 00 

1. 000. 000, 00 








125% ~ 
图 7-12 格式 化 数字 字符 串 
疑问 2: 如 何 删除 由 addslashes0) 函 数 添 加 的 反 斜 杠 ? 


答 : 在 PHP 中 ， 如 果 想 删除 addslashes() 函数 添加 的 反 斜 枉 ， 需 要 使 用 stripslashes(O) 函 
该 函数 可 用 于 清理 从 数据 库 中 或 者 从 HTML 表单 中 取 回 的 数据 。 

例如 以 下 代码 : 

<?php 

echo stripslashes(" 数 点 \' 雨 声 风 约 住 。 滕 肛 淡 月 云 来 去 。") ; 


人 


程序 运行 结果 如 图 7-13 所 示 。 
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= 日 Xx 
[B+ |@ httpi//ocalho.. P ~- © | Blocalhost x 
妨 编 (E) ”查看 (V) 收藏 夫 (A) 工具 (T) 帮助 (H) 
数 点 ' 雨 声 风 约 住 。 滕 胱 淡 月 云 来 去 。 
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第 8 章 
匹配 文本 有 妙招 
一 正则 表达 式 


上 一 章 介绍 的 对 字符 串 的 处 理 比较 简单 ， 只 是 使 用 一 定 的 函数 对 字符 串 进行 处 
理 ， 无 法 满足 对 字符 串 进行 复杂 处 理 的 需求 。 此 时 就 需要 使 用 正则 表达 式 。 本 章 重 
点 学 习 正 则 表达 式 的 使 用 方法 和 技巧 。 
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8.1 什么 是 正则 表达 式 


正则 表达 式 是 把 文本 或 字符 串 按照 一 定 的 规范 或 模型 表示 的 方法 ， 经 常用 于 文本 的 匹配 
操作 。 

例如 ， 验 证 用 户 在 线 输 入 的 邮件 地 址 的 格式 是 否 正确 时 ， 常 常 使 用 正则 表达 式 技术 来 匹 
配 ， 若 匹配 ， 则 用 户 所 填写 的 表单 信息 将 会 被 正常 处 理 ; 反之 ， 如 果 用 户 输入 的 邮件 地 址 与 
正则 表达 的 模式 不 匹配 ， 将 会 弹出 提示 信息 ， 要 求 用 户 重新 输入 正确 的 邮件 地 址 。 可 见 正则 
表达 式 在 Web 应 用 的 逻辑 判断 中 具有 举足轻重 的 作用 。 


8.2 正则 表达 式 的 语法 规则 


一 般 情况 下 ， 正 则 表达 式 由 两 个 部 分 组 成 ， 分 别 是 元 字符 和 文本 字符 。 元 字符 就 是 具有 
特殊 含义 的 字符 ， 如 “?” 和 “*” 等 ， 文 本 字符 就 是 普通 的 文本 ， 如 字母 、 数 字 等 。 下 面 主 
要 讲述 正则 表达 式 的 语法 规则 。 


8.2.1 方 括号 ([ ]) 


方 括号 内 的 一 串 字 符 是 将 要 用 来 进行 匹配 的 字符 。 

例如 ， 正 则 表达 式 在 方 括号 内 的 [name] 是 指 在 目标 字符 串 中 寻找 字母 n、a、m、e。[jk] 表 
示 在 目标 字符 串 中 寻找 字符 j 和 kk。 
8.2.2” 连 字符 (-) 

在 很 多 情况 下 ， 不 可 能 逐个 列 出 所 有 的 字符 。 比 如 ， 若 需要 匹配 所 有 英文 字符 ， 则 把 26 
个 英文 字母 全 部 输入 ， 这 会 十 分 困难 。 这 样 ， 就 有 了 如 下 表示 。 

(1) [a-z]: 表示 匹配 英文 小 写 从 a~z 的 任意 字符 。 

(2) [A-Z]: 表示 匹配 英文 大 写 从 A 一 Z 的 任意 字符 。 

(3) [A-Za-z]: 表示 匹配 英文 大 小 写 从 大 写 A 到 小 写 z 的 任意 字符 。 

(4) [0-9]: 表示 匹配 从 0 一 9 的 任意 十 进 制 数 。 

由 于 字母 和 数字 的 区 间 固 定 ， 所 以 根据 这 样 的 表示 方法 [开始 -结束 ]， 程 序 员 可 以 重新 定 
义 区 间 大 小 ， 如 [2-7]、[c- 旨 等 。 


8.2.3 点 号 字符 (.) 


点 号 字符 在 正则 表达 式 中 是 一 个 通配符 ， 它 代表 所 有 字符 和 数字 。 例 如 ，“.er” 表 示 所 
有 以 er 结尾 的 3 个 字符 的 字符 串 ， 可 以 是 per、ser、ter、@er、&er 等 。 


8.2.4 ”限定 符 (+*?{n,m}) 
加 号 “+” 表示 其 前 面 的 字符 至 少 有 一 个 。 例 如 ，“9+” 表 示 目 标 字符 串 包 含 至 少 一 个 9。 











星 号 “* ”表示 其 前 面 的 字符 有 不 止 一 个 或 0 个 。 例 如 ，“y*” 表 示 目 标 字 符 串 包含 0 个 
或 者 不 止 一 个 y。 

问号 “?” 表 示 其 前 面 的 字符 有 一 个 或 0 个 。 例 如 ，“y?” 表 示 目 标 字 符 串 包含 0 个 或 者 
一 个 ye 

大 括号 “ {n,m} ”表示 其 前 面 的 字符 有 n 或 m 个 。 例 如 ，“a{3,5}” 表 示 目 标 字符 串 包含 
3 个 或 者 5 个 a， 而 “a{3}” 表 示 目 标 字 符 串 包含 3 个 a，“a{3,}” 表 示 目 标 字符 串 包含 至 少 
了 全 ae 


点 号 和 星 号 一 起 使 用 ， 表 示 广 义 匹配 。 即 “.* ”表示 匹配 任意 字符 。 
8.2.5 行 定位 符 (^ 人 和 9$) 


行 定 位 符 用 来 确定 匹配 字符 串 所 要 出 现 的 位 置 。 

如 果 是 在 目标 字符 串 开 头 出 现 ， 则 使 用 符号 “^”; 如 果 是 在 目标 字符 串 结尾 出 现 ， 则 使 
用 符号 “$”。 例 如 ，^xiaoming 是 指 “xiaoming” 只 能 出 现在 目标 字符 串 开头 。8895$ 是 指 
“8895” 只 能 出 现在 目标 字符 串 结尾 。 

有 一 个 特殊 表示 ， 即 同时 使 用 和 $ 两 个 符号 ， 就 是 “^[a-zl]$”， 表 示 目 标 字符 串 只 包含 
从 a 一 z 的 单个 字符 。 
8.2.6 ”排除 字符 ([A]) 

符号 “^” 在 方 括号 内 所 代表 的 意义 则 完全 不 同 ， 它 表示 一 个 逻辑 “和 否 ”， 排 除 匹配 字符 串 
在 目标 字符 串 中 出 现 的 可 能 。 例 如 ，[^0-9] 表 示 目 标 字 符 串 包 含 从 0 一 9 以 外 的 任意 其 他 字符 。 
8.2.7 ”括号 字符 (()) 


括号 字符 (0) 表 示 子 串 ， 所 有 对 包含 在 子 串 内 字符 的 操作 ， 都 是 以 子 串 为 整体 进行 的 ， 也 
是 把 正则 表达 式 分 成 不 同 部 分 的 操作 符 。 


8.2.8 选择 字符 (|) 
选择 字符 (|) 表 示 “ 或 ”选择 。 例 如 ，“comlenlcom.cnlnet” 表 示 目 标 字符 串 包含 com 或 cn 


或 com.cn 或 net。 


8.2.9 ” 转 义 字 符 与 反 斜 杠 


由 于 “\” 在 正则 表达 式 中 属于 特殊 字符 ， 所 以 ， 如 果 单 独 使 用 此 字符 ， 则 将 直接 表示 为 
作为 特殊 字符 的 转 义 字符 。 如 果 要 表示 反 斜 杠 字符 本 身 ， 则 应 当 在 此 字符 的 前 面 添 加 转 义 字 
符 “\” ， 即 为 “\” 。 


8.2.10 认证 E-mail 的 正则 表达 式 
在 处 理 表单 数据 的 时 候 ， 对 用 户 的 E-mail 进行 认证 是 十 分 常用 的 。 如 何 判断 用 户 输入 的 
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是 一 个 E-mail 地 址 呢 ? 就 是 用 正则 表达 式 来 匹配 。 其 语法 格式 如 下 : 


^[RA-Za-z0-9_-.]+e[R-Za-Z0-9_]+N- [A-2a-z0-9.]+$ 


其 中 ^[A-Za-z0-9_.]+ 表 示 至 少 有 一 个 英文 大 小 写字 符 、 数 字 、 下 划 线 、 点 号 ， 或 者 这 些 字 
符 的 组 合 。 

@ 表 示 E-mail 中 的 “@”。 

[A-Za-z0-9_]+ 表 示人 至 少 有 一 个 英文 大 小 写字 符 、 数 字 、 下 划 线 ， 或 者 这 些 字符 的 组 合 。 

\ 表 示 E-mail 中 “.com” 之 类 的 点 。 这 里 点 号 只 是 点 本 身 ， 所 以 用 反 斜 杠 对 它 进 行 转 义 。 

[A-Za-z0-9.]+$ 表 示 至 少 有 一 个 英文 大 小 写字 符 、 数 字 、 点 号 ， 或 者 这 些 字符 的 组 合 ， 并 
且 直 到 这 个 字符 串 的 末尾 。 


8.3 ”Perl 兼容 正则 表达 式 函 数 


在 PHP 中 有 两 类 正则 表达 式 函 数 : 一 是 Perl 兼容 正则 表达 式 函 数 ， 二 是 POSIX 扩展 正 
则 表达 式 函 数 。 二 者 差别 不 大 ， 推 荐 使 用 Perl 兼容 正则 表达 式 函 数 。 因 此 ， 下 面 都 是 以 Perl 
兼容 正则 表达 式 函 数 为 例 进行 说 明 的 。 


8.3.1 使 用 正则 表达 式 对 字符 串 进 行 匹 配 


用 正则 表达 式 对 目标 字符 串 进行 匹配 是 正则 表达 式 的 主要 功能 。 

完成 这 个 操作 需要 用 到 preg_match() 函 数 。 这 个 函数 是 在 目标 字符 串 中 寻找 符合 特定 正则 
表达 规范 的 字符 串 子 串 。 根 据 指定 的 模式 来 匹配 文件 名 或 字符 串 。 它 的 语法 格式 如 下 : 

preg_match (正则 表达 式 ， 目 标 字 符 串 , [ 数组 ] ) 


其 中 数组 为 可 选 参数 ， 存 储 匹配 结果 的 数组 。 
下 面 通过 例子 介绍 利用 preg_match0 函 数 匹 配 字符 串 的 方法 和 技巧 。 
【 例 8.1】 使 用 preg_match0 函 数 ( 示 例文 件 ch08\8.1.php)。 


<?php 
$aa = "When you are old and grey and full of sleep"; 
$bb = "人 生 若 只 如 初 见 ， 何 事 秋风 翡 画 扇 "， 
Sre = "/when/™; // 区 分 大 小 写 
$re2 = "/when/i"; // 不 区 分 大 小 写 
$re3 = "/ 何 事 /"; 
if(preg match($re，S$aa，$a)){  ”// 第 1 次 匹配 时 区 分 大 小 写 
echo "第 1 次 匹配 结果 为 : "; 
print r($a); 
echo "<br/>"; 
1 
ifE(preg_match (Sre2，S$aa，S$b)){ // 第 2 次 匹配 时 不 区 分 大 小 写 
echo "第 2 次 匹配 结果 为 : "; 
Print_r(Sb) 
echo "<br/>"; 
3 
if(preg match($re3，$bb，$c)){ // 第 3 次 匹配 中 文 


echo "第 3 次 匹配 结果 为 : "; 
print r($c); 
3 








?> 
程序 运行 结果 如 图 8-1 所 示 。 一 一 二 到 
【案例 剖析 】 OE 








(1) $aa 就 是 一 个 完整 的 字符 串 ， 用 $re 这 个 正则 第 次 下 本 结果 为 Array ( [0] = 何事 ) 
规范 ， 由 于 不 区 分 大 小 写 ， 所 以 第 1 次 匹配 没 结果 。 WE 
(2) 第 2 次 匹配 不 再 区 分 大 小 写 ， 将 匹配 的 子 串 储 8-1 使 用 preg_match() 函 数 
存在 名 为 $a 的 数组 中 。print r($a) 打 印 数组 ， 得 第 1 行 
数组 的 输出 。 
(3) 第 3 次 匹配 为 中 文 匹 配 ， 结 果 匹 配 成 功 ， 得 到 相应 的 输出 。 
preg_match() 第 1 次 匹配 成 功 后 就 会 停止 匹配 ， 如 果 要 实现 全 部 结果 的 匹配 ， 即 搜索 到 
字符 串 结尾 处 ， 则 需要 使 用 preg_match_ all0 函 数 。 
【 例 8.2】 使 用 preg_match_all0 函 数 (示例 文件 ch08\8.2.php)。 


第 2 次 匹配 结果 为 : Array ( [0] -> When ) | 








站 降 袖 旦 同一 一 湛 导 亿 讨 准 副 则 翰 8 游 鲁 





<?php 

$aa = "When you are old and grey and full of sleep"; 

$bb = "人 生 若 只 如 初 见 ， 何 事 秋风 翡 画 扇 。 人 生 若 只 如 初 见 ， 何事 秋风 翡 画 户 。 ey 
$re = "/And/™"; // 区 分 大 小 写 


$re2 = "/And/i"; // 不 区 分 大 小 写 
$re3 = "/ 何 事 /"; 
if(preg match all($re，$aa，$a)){  ”// 第 1 次 匹配 时 区 分 大 小 写 
echo "第 1 次 匹配 结果 为 :"; 
print_r($a); 
echo "<br/>"; 
if(preg_match_all($re2，$aa，$b)){  // 第 2 次 匹配 时 不 区 分 大 小 写 
echo "第 2 次 匹配 结果 为 :"; 
print_r ($b); 
echo "<br/>"; 
F 
if(preg_match_all($re3，S$bb，$c)){ // 第 3 次 匹配 中 文 
echo "第 3 次 匹配 结果 为 :"; 


print _r($c); 





} 


> 
程序 运行 结果 如 图 8-2 所 示 。 从 结果 可 以 看 出 ，preg_match_all0 函数 匹配 了 所 有 的 结果 。 
@ EB hep /MocahowWcoie/dH0E PD - © | localhon 辐 ne 





第 2 次 匹配 结果 为 : Array ( [0] =>》 Array ( [0] => and [1] => and ) ) 
第 3 次 匹配 结果 为 :Array ( [0] => Array ( [0] => 何事 [1] =》 何事 ) ) 


2 





8-2 使 用 preg_match_all() 函 数 
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8.3.2 使 用 正则 表达 式 蔡 换 字符 串 的 子 串 


做 好 了 字符 上 串 及 其 子 串 的 匹配 ， 如 果 需 要 对 字符 串 的 子 串 进行 蔡 换 ， 可 以 使 用 
preg_replace() 函 数 来 完成 。 其 语法 格式 如 下 : 
preg_replace (正则 表达 式 规范 ， 欲 取代 字符 串 子 串 ， 目 标 字 符 串 ，[ 蔡 换 的 个 数 ] ) 


如 果 省 略 蔡 换 的 个 数 或 者 蔡 换 的 个 数 为 -1， 则 所 有 的 匹配 项 都 会 被 替换 。 
下 面 通过 例子 介绍 利用 正则 表达 式 取代 字符 串 子 串 的 方法 和 技巧 。 
【 例 8.3】 使 用 preg_replace0 函 数 ( 示 例文 件 ch08\8.3.php)。 


<?php 

$aa = "When you are old and grey and full of sleep"; 

$bb = "人 生 若 只 如 初 见 ， 何 事 秋风 莫 画 扇 。 人 生 若 只 如 初 见 ， 何 事 秋风 悲 画 扇 。"; 
$aa= preg_replace('/\s/','-',$aa); 

echo "第 1 次 替换 结果 为 : "."<br/>"; 

echo $aa."<br/>"; 

$bb= preg_replace (' /何事 /',' 往 事 ', $bb); 

echo "第 2 次 普 换 结果 为 : "."<br/>"; 

echo $bb; 

> 





程序 运行 结果 如 图 8-3 所 示 。 
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第 1 次 普 换 结 果 为 : 
yo 





8-3 ”使 用 正则 表达 式 蔡 换 字符 串 的 子 串 
【案例 剖析 】 


(1) 第 一 次 替换 是 将 空格 蔡 换 为 “-”， 然 后 将 蔡 换 后 的 结果 输出 。 
(2) 第 二 次 替换 是 将 “何事 ”替换 为 “往事 ”， 然 后 将 替换 后 的 结果 输出 。 


8.3.3 使 用 正则 表达 式 切 分 字符 串 


使 用 正则 表达 式 可 以 把 目标 字符 串 按 照 一 定 的 正则 规范 切 分 成 不 同 的 子 串 。 完 成 此 操作 
需要 使 用 到 strtok0 函 数 。 它 的 语法 格式 如 下 : 
strtok (正则 表达 式 规范 ， 目 标 字 符 串 ) 
这 个 函数 是 指 以 正则 表达 式 规范 内 出 现 的 字符 为 准 ， 把 目标 字符 串 切 分 成 若干 个 子 串 ， 
并 且 存 入 数组 。 
下 面 通 过 例子 介绍 利用 正则 表达 式 切 分 字符 串 的 方法 和 技巧 。 
【 例 8.4】 使 用 strtok0 函 数 (示例 文件 ch08\8.4.php)。 
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<?php 

$string = "Hello world. Beautiful day today."; 
$token = strtok($string, ™ "™); 

while ($token !== false) 

: 

echo "$token<br />"; 

$token = strtok(" ")» 


> 


程序 运行 结果 如 图 8-4 所 示 。 





二 (a! x 
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Hello 
world. 
Beautiful 
day 

today. 
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图 8-4 利用 正则 表达 式 切 分 字符 串 





【案例 剖析 】 


(1) $string 为 包含 多 种 字符 的 字符 串 。strtok($string，" ") 对 其 进行 切 分 ， 并 将 结果 存 入 数 
组 $token。 


(2) 其 正则 规范 为 " "， 是 指 按 空 格 将 字符 串 切 分 。 


8.4 ”案例 实战 一 一 创建 商品 在 线 订单 页 面 


本 例 主要 创建 商品 采购 系统 的 在 线 订购 商品 页 面 。 具 体操 作 步 又 如 下 。 
在 网 站 主 目录 下 建立 文件 caigouphp。 代 码 如 下 : 


<!DOCTYPE html> 

<html> 

<head> 

<meta http-equiv="Content-Type" content="text/html; charset=gb2312" 
您 的 商品 订单 信息 : 

</head> 

<body> 

<?php 

$DOCUMENT_ROOT = $_SERVER['DOCUMENT ROOT']; 
$customername = trim($ POST['customername']); 
S$gender = $_ POST['gender']; 

$arrivaltime = $_POST['arrivaltime']; 

S$phone = trim($_POST['phone']); 

$email = trim($_POST['email']); 

$info = trim($ POST['info']); 


$rel = "/^\w+@\w+\.comlcnlnet$/"; // 不 区 分 大 小 写 
$re2= "/^1[34578] \d{9}$/"; 


> 
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IfE(!preg_match (S$relvSemail) ){ 
echo "这 不 是 一 个 有 效 的 email 地 址 ， 请 返回 上 页 上 且 重 试 "; 
exit; 
3 
if(!preg match($re2, $phone)){ 
echo "这 不 是 一 个 有 效 的 电话 号 码 ， 请 返回 上 页 且 重 试 "; 
exit; 
是 
if($gender == "m"){ 
$customer = "先生 "; 
J}elsef{ 
$customer = "女士 "; 


于 

echo '<p> 您 的 商品 信息 已 经 上 传 ， 我 们 正在 为 您 备货 。 确认 您 的 商品 订单 信息 如 下 :</p>'; 
echo $customername."\t".$customer.' 您 好 ! '.， 取 货 时 间 为 : ' .$arrivaltime.' 天 
后 。 您 的 电话 为 ' .$phone."。 我 们 将 会 发 送 一 封 电子 邮件 到 您 的 email 邮箱 : " .Semail."。<br 
/><br/> 另 外 ， 我 们 已 经 确认 了 您 的 商品 采购 信息 : <br /><br />"; 

echo nl2br ($info); 

echo "<p> 您 的 商品 采购 时 间 为 :" .date('Y m d H: i: s')."</p>"; 

</body> 

</html> 


在 网 站 主 目录 下 建立 文件 shangpin.html， 代 码 如 下 : 


<!DOCTYPE html> 
<html> 
<head> 
<h2>BBSS 商品 采购 系统 </h2> 
</HEAD> 
<BODY> 
<form action="caigou.php" method="post"> 
<table> 
<tr bgcolor="#3399FF"> 
<td> 客 户 姓 名 : </td> 
<td><input type="text" name="customername" size="20" /></td> 
</tr> 
<tr bgcolor="#CCCCCC"> 
<td> 客 户 性 别 : </td> 
<td> 
<select name="gender"> 
<option value="m"> 男 </option> 
<option value="f"> 女 </option> 
</select> 
</td> 
EP 
<tr bgcolor="#3399FF"> 
<td> 取 货 时 间 :</td> 
<td> 
<select name="arrivaltime"> 
<option value="1"> 当 天 </option> 
<option value="1">1 天 后 </option> 
<option value="2">2 天 后 </option> 
<option value="3">3 天 后 </option> 
<option value="4">4 天 后 </option> 





gs 
忆 
第 [ 
<option value="5"> 协 商 时 间 </option> oo， 
</select> 章 | 
</td> 
he 加 
<tr bgcolor="#CCCCCC"> 文 
<td> 电 话 :</td> 本 
<td><input type="text" name="phone" size="20" /></td> a 
</tr> 
<tr bgcolor="#3399FF"> 
<td>email:</td> 
<td><input type="text" name="email" size="30" /></td> 正 
</tr> 则 
<tr bgcolor="#CCCCCC"> 要 
<td> 商 品 采购 信息 :</td> 起 
<td> < 


<textarea name="info" rows="10" cols="30"> 商 品 采购 的 具体 信息 ， 请 填 在 这 里 。 
</textarea> 
</td> 
eHEr% 
<tr bgcolor="#666666"> 
<td align="center"><input type="submit" value=" 确 认 商 品 采购 订单 /></td> 
</Er> 
</table> 
</form> 
</body> 
</html> 


ES) 运行 shangpin html， 结 果 如 图 8-5 所 示 。 
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图 8-5 shangpin.html 的 运行 结果 
EEJz 填写 表单 。“ 客 户 姓 名 ”为 “王小明 ”，“ 性 别 ” 为 “ 女 ”，“ 取 货 时 间 ” 为 
“1 天 后 ”，“ 电 话 ” 为 “13012345678”，E-mail 为 123456789@qq.com，“ 商 品 
采购 信息 ”为 “1. 冰箱 203 号 6 台 ; 2. 洗衣 机 115 号 7 台 ; 3. 空调 165 号 5 台 ; 4. 
电视 机 169 号 15 台 ”。 单 击 “ 确 认 商 品 采购 订单 ”按钮 ， 浏 览 器 会 自动 跳 转 至 
caigou.php 页 面 ， 显 示 如 图 8-6 所 示 的 结果 。 
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您 的 商品 订单 信息 : 
您 的 商品 信息 已 经 上 传 ， 我 们 正在 为 您 备货 。 确认 您 的 商品 订单 信息 如 下 : 








王小明 女士 您 好 ! 取 货 时 间 为 。 1 天 后 。 您 的 电话 为 13012345678。 我 
们 将 会 发 送 一 封 电子 邮件 到 您 的 email 邮 箱 : 123456789@qq. com- 





另外 ， 我 们 已 经 确认 了 您 的 商品 采购 信息 : 
1. 冰箱 203 与 6 台 

2. 洗衣 机 115 号 7 台 

3， 空 调 165 号 5 台 

4 电视 机 169 号 15 台 


您 的 商品 采购 时 间 为 :2017 08 04 04: 53: 56 
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8-6 提交 后 的 显示 结果 
【案例 剖析 】 


(1) “ $customername=trim($ POST['customername']): $phone = trim($ POST['phone']): 
Semail = trim($_ POST['email]): $info = trim($_ POST['info]): ”都 是 通过 文本 框 直接 输入 的 。 所 
以 ， 为 了 保证 输入 字符 串 的 整洁 ， 以 方便 处 理 ， 则 需要 使 用 trim0， 来 对 字符 串 的 前 后 空格 进 
行 清除 。 另 外 ，ltrim0) 清 除 左 边 的 空格 ，rtrim() 清 除 右边 的 空格 。 

(2) “Srel = "/ 人 w+@\Wwt\.comlcnlnet$/";” 中 规定 了 判断 邮箱 是 否 合 规 的 正则 表达 式 。 

(3) “Sre2= "A1[34578]\d{9}$/";” 中 规定 了 判断 手机 号 是 否 合 规 的 正则 表达 式 。 

(4) 由 于 要 显示 中 文 ， 需 要 对 文字 编码 进行 设置 ，charset=gb2312， 就 是 简体 中 文 的 文字 
编码 。 


8.5 疑难 解 惑 


疑问 1: 在 PHP 中 正则 表达 式 的 语法 有 几 种 


答 : 在 PHP 的 正则 表达 式 的 语法 中 ， 一 种 是 POSIX 语法 ， 另 一 种 是 Perl 语法 。POSIX 
语法 是 先前 所 介绍 的 语法 。Perl 语法 则 不 同 于 POSIX 语法 。Perl 语法 的 正则 表达 式 是 以 “/” 
开头 和 以 “/” 结 尾 的 ， 如 “/name/” 便 是 一 个 Perl 语法 形式 的 正则 表达 式 。 


疑问 2: 在 PHP7 中 被 舍弃 的 正则 表达 式 函 数 有 哪些 ? 有 没有 新 的 函数 替代 ? 


答 : 读者 在 查看 一 些 早期 的 源 代码 时 ， 会 发 现 一 些 被 舍弃 的 正则 表达 式 。 下 面 介绍 哪些 
函数 被 舍弃 ， 并 介绍 新 的 蔡 代 函数 。 

(1) ereg0。 该 函数 已 经 被 舍弃 ， 使 用 新 的 函数 preg_match() 蔡 代 。 

(2) ereg_replace()。 该 函数 已 经 被 舍弃 ， 使 用 新 的 函数 preg_ replaceO0 蔡 代 。 

(3) eregi()。 该 函数 已 经 被 舍弃 ， 使 用 新 的 函数 preg_match0 配 合 i 修正 符 蔡 代 。 

(4) eregi_replace()。 该 函数 已 经 被 舍弃 ， 使 用 新 的 函数 preg replace0 配 合 i 修正 符 蔡 代 。 

(5) split0。 该 函数 已 经 被 舍弃 ， 使 用 新 的 函数 preg_split0 替 代 。 

(6) spliti0。 该 函数 已 经 被 舍弃 ， 使 用 新 的 函数 preg_splitO 配 合 i 修正 符 替 代 。 


第 9 章 
表单 的 动态 效果 
一 一 PHP 与 Web 

页 面 交互 


PHP 是 一 种 专门 用 于 Web 开发 的 服务 器 端 脚本 语言 。 从 这 个 描述 可 以 知道 ， 
PHP 要 打交道 的 对 象 主要 有 服务 器 (Server) 和 基于 Web 的 HIML( 超 文本 标识 语 
言 )。 在 使 用 PHP 处 理 Web 应 用 时 ， 需 要 把 PHP 代码 谈 入 到 HTML 文件 中 。 每 次 
当 这 个 HTML 网 页 被 访问 的 时 候 ， 其 中 幅 入 的 PHP 代码 就 会 被 执行 ， 并 且 返 回 给 
请 求 浏览 器 已 生成 好 的 HIML。 换 名 话说 ， 在 上 述 过 程 中 ，PHP 就 是 用 来 执行 且 生 
成 HTML 的 。 本 章 主要 讲述 PHP 与 Web 页 面 的 交互 操作 技术 。 
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9.1 创建 动态 内 容 


为 什么 要 使 用 动态 内 容 呢 ? 因为 动态 内 容 可 以 给 网 站 使 用 者 不 同 的 和 实时 变化 的 内 容 。 
极 大 地 提高 了 网 站 的 可 用 性 。 如 果 Web 应 用 都 只 是 使 用 静态 内 容 ， 则 Web 编程 就 完全 不 用 
引入 PHP、JSP 和 ASP 等 服务 器 端 脚本 语言 了 。 通 俗 地 说 ， 使 用 PHP 语言 的 主要 原因 之 一 ， 
就 是 要 产生 动态 内 容 。 

下 面 进 行使 用 动态 内 容 案 例 的 讲解 。 此 例 中 ， 在 先 不 涉及 变量 和 数据 类 型 的 情况 下 ， 仅 
使 用 PHP 中 的 一 个 内 置 函 数 来 获得 动态 内 容 。 此 动态 内 容 就 是 使 用 date0 函 数 获得 Web 服务 
器 的 时 间 。 

【 例 9.1】 使 用 date0 函 数 (示例 文件 ch09\9.1.php)。 


<?php 

date default timezone set ("PRC"); 
echo "现在 的 时 间 为 : "; 

echo date("H:i:s Y m d"); 

?> 


程序 运行 结果 如 图 9-1 所 示 。 过 一 段 时 间 ， 再 次 运行 上 述 PHP 页 面 ， 即 可 看 到 显示 的 内 
容 发 生 了 动态 变化 ， 如 图 9-2 所 示 。 
二 口 x 
[@@- |G hep/ocalho.. P - © | Blocalhost 
文件 (有 ” 忽 辑 (E】 查看 (V) 收藏 交 (A) 工具 (T) 帮助 (H) 
现在 的 时 间 为 : 16:11:24 2017 08 04 | 


- 0O x 
@ OG httpi//localho., P © | @ localhost 
文件 (六 多 (E) 得 看 (V) 收藏 交 (A) 工具 (T) 帮助 (H) 


现在 的 时 间 为 : 16:12:10 2017 08 04 


























125% ~ 胞 125% > 











图 9-1 初始 结果 图 9-2 时 间 发 生 了 变化 


【案例 训 析 】 

(1) 页 面 上 的 “现在 的 时 间 为 : 16:11:24 2017 08 04” 是 由 <?php echo "现在 的 时 间 为 :"; 
echo date("H:i:s Ym d"); ?> 生成 的 。 

(2) 由 于 时 间 是 由 date0 函 数 动态 生成 并 且 实 时 更 新 的 ， 所 以 再 次 打开 或 刷新 此 文件 的 时 
候 ，PHP 代码 再 次 执行 ， 所 输出 的 时 间 也 会 发 生 改变 。 

(3) 此 例 通 过 dateO) 函 数 处 理 系统 时 间 ， 得 到 动态 内 容 。 时 间 处 理 是 PHP 中 的 一 项 重要 
功能 。 


9.2 ”表单 与 PHP 


不 管 是 一 般 的 企业 网 站 还 是 复杂 的 网 络 应 用 ， 都 离 不 开 数 据 的 添加 。 通 过 PHP 服务 器 端 
脚本 语言 ， 程 序 可 以 处 理 那些 通过 浏览 器 对 Web 应 用 进行 数据 调用 或 添加 的 请 求 。 


回忆 一 下 平常 使 用 的 网 站 数据 输入 功能 ， 不 管 是 Web 邮箱 ， 还 是 QQ 留言 ， 都 经 常 要 填 
写 一 些 表格 ， 再 由 这 些 表格 把 数据 发 送出 去 。 而 完成 这 个 工作 的 部 件 就 是 “表单 (form)”。 

虽然 表单 是 HTML 语言 中 的 ， 但 是 PHP 与 form 变量 的 衔接 是 无 颖 的 。PHP 关心 的 是 怎 
样 获得 和 使 用 form 中 的 数据 。PHP 功能 强大 ， 可 以 轻松 地 对 表单 进行 处 理 。 

处 理 表单 数据 的 基本 过 程 是 : 数据 从 Web 表单 发 送 到 PHP 代码 ， 经 过 处 理 ， 再 生成 
HTML 输出 。 它 的 处 理 原 理 是 : 当 PHP 处 理 一 个 页 面 的 时 候 ， 会 检查 URL、 表 单数 据 、 上 传 
文件 、 可 用 cookie、Web 服务 器 和 环境 变量 。 如 果 有 可 用 信息 ， 就 可 以 通过 PHP 访问 自动 全 
局 变量 数组 $ GET、$_ POST、$_FILES、$_COOKIE、$_SERVER、$_ENV 得 到 。 





9.3 设计 表单 元 素 


表单 是 一 个 比较 特殊 的 组 件 。 在 HIML 中 有 着 比较 特殊 的 功能 和 结构 。 下 面 介绍 表单 的 
一 些 基本 元 素 。 


9.3.1 表单 的 基本 结构 
表单 的 基本 结构 是 由 <fomm></fomm> 标 识 包 训 的 区 域 。 例 如 : 


<HTML> 
<HEAD> 
</HEAD> 
<BODY> 
<form action=" " method=" " enctype=" "> 


回头 司 昼 qe 人 ddHd 一 一 宁海 寺 过 可 地 洒 届 6 小 鲁 





ee 

</BODY> 

</HTML> 

其 中 ，<form> 标 识 内 必须 包含 属性 。action 指定 数据 所 要 发 送 到 的 对 象 文件 。method 指 
定数 据 传输 的 方式 。 

如 果 在 上 传 文件 等 操作 时 ， 还 要 定义 enctype 属性 来 指定 数据 类 型 。 


9.3.2 文本 框 


文本 框 是 form 输入 框 中 最 为 常见 的 一 个 。 下 面 通过 例子 来 讲述 文本 框 的 使 用 方法 。 
具体 操作 步骤 如 下 。 
EE 在 网 站 根 目录 下 创建 文件 9.2.html。 代 码 如 下 : 


<html> 

<head></head> 

<body> 

<form action="9.2.php" method="post"> 
<h3> 输 入 一 个 信息 (比如 名 称 ) : </h3> 

<input type="text" name="name" size="]10" /> 
</form> 
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</body> 
</html> 


EDPD 创建 文件 9.2.php。 代 码 如 下 : 

<?php 

Sname = $ POST['name']; 

echo $name; 

人 

运行 9.2.html， 结 果 如 图 9-3 所 示 。 在 文本 框 中 输入 “冰箱 ”， 按 Enter 键 确 认 ， 结 果 如 
图 9-4 所 示 。 





医 二 1 加 甘 


























OTs-e i @ [@@- Grep/ecaho.. P - & 
文员 中。 名 加 (6 喜 看 (V)。 收 应 灾 从 ) 工具 部 0 文帝 各 (E) 坦 看 (V) 收藏 次 (A) 工 ” 
答 入 一 个 信息 (比如 名 称 ) : 
讨 条 
用 125% R125% ~ 
图 9-3 使 用 文本 框 图 9-4 ”程序 运行 结果 
【 业 例 剖析 】 
(1) <input type="text" name="name" size="10" /> 语句 定义 了 form 的 文本 框 。 定 义 一 个 输 
入 框 为 文本 框 的 必要 因素 为 : 


<input type="text" ... /> 


这 样 就 定义 了 一 个 文本 框 ， 其 他 属性 则 如 例 中 一 样 ， 可 以 定义 文本 框 的 name 属性 ， 以 确 
认 此 文本 框 的 唯一 性 ， 定 义 size 属性 以 确认 此 文本 框 的 长 度 。 
(2) 在 9.2.php 文件 中 ， 则 使 用 了 文本 框 的 name 值 “name”， 输 出 文本 框 中 的 内 容 。 


9.3.3” 复 选 框 


复 选 框 可 用 于 选择 一 项 或 者 多 项 。 下 面 通过 案例 详细 讲解 PHP 中 复 选 框 的 使 用 方法 。 具 
体操 作 步 又 如 下 。 
EC 创建 文件 9.3.html。 代 码 如 下 : 


<html> 

<head></head> 

<body> 

<form action="9.3.php" method="post"> 

<h3> 确 认 此 项 (可 复 选 ) : </h3> 

<input type="checkbox" name="achecked" checked="checked" value="1" /> 
选择 此 项 传递 的 A 项 的 value 值 。 

<input type="checkbox" name="bchecked" value="2" /> 
选择 此 项 传递 的 B 项 的 value 值 。 

<input type="checkbox" name="cchecked" value="3" /> 
选择 此 项 传递 的 c 项 的 value 值 。 


</form> 
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</body> 
</html> 


创建 文件 9.3.php。 代 码 如 下 : 


<?php 

$name = $_POST['name']; 
if(isset($ POsST['achecked'])){ 
$achecked = $ POST['achecked']; 
} 

if(isset($ POST['bchecked'])){ 
$bchecked = $_ POST['bchecked']; 
E 

if(isset($ POST['cchecked'])){ 
$cchecked = $_POST['cchecked']; 
上 


if(isset(S$achecked) and $achecked == 1)1{ 

echo "选项 A 的 value 值 已 经 被 正确 传递 。<br />"; 

}elsel{ 

echo "选项 A 没有 被 选择 ， 其 value 值 没有 被 传递 。<br />"; 
} 

if(isset(S$bchecked) and $bchecked == 2){ 

echo "选项 B 的 value 值 已 经 被 正确 传递 。<br />"; 

}else{ 

echo "选项 B 没有 被 选择 ， 其 value 值 没有 被 传递 。<br />"; 
j 

if(isset($cchecked) and $cchecked == 3){ 

echo "选项 c 的 value 值 已 经 被 正确 传递 。<br />"; 

}elsei 

echo "选项 c 没有 被 选择 ， 其 value 值 没有 被 传递 。<br />"; 
echo $name."<br/>"; 

汉学 


EEJe 运行 9.3.html， 结 果 如 图 9-5 所 示 。 色 选 复 选 框 并 在 文本 框 中 输入 内 容 后 ， 按 


Enter 键 确认 ， 结 果 如 图 9-6 所 示 。 








| [on I I Qee 
文件 旧 。 演 缀 E) 查看 fV) 必 营 幸 (A) 工具 (0 者 赎 (H) 
确认 此 项 (可 复 选 ): 
- 
加 选择 此 项 传递 的 A 项 的 value 值 . 回 选择 此 项 传递 的 B 项 的 value 值 . [让 本 
站 地 此 项 传 的 项 的 value 信 ， 人 © | 
| 选项 的 value 值 已 经 被 正确 传递 。 
[fa 选项 8 的 value 什 已 经 被 正确 传递. 
| 选项 C 没 有 被 选择 ， 其 value 值 没有 被 传递 
洗衣 机 
125% > 胞 125% 
图 9-5 使 用 复 选 框 图 9-6 程序 运行 结果 





























【案例 剖析 】 


(1) <input type="checkbox" name="inputchecked" checked="checked" value="1" /> 语句 定义 
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了 复 选 框 。 定 义 一 个 input 标识 为 复 选 框 时 需要 指定 类 型 为 checkbox: 


<input type="checkbox" ... /> 


定义 为 复 选 框 之 后 ， 还 需要 定义 复 选 框 的 name 属性， 以 确定 在 服务 器 端 程序 的 唯一 性 。 
定义 value 属性 ， 以 确定 此 复 选 框 所 要 传递 的 值 。 定 义 checked 属性 ， 以 确定 复 选 框 的 默认 状 
态 ， 若 为 checked， 则 是 默认 为 选中 ;如 果 不 定义 此 项 ， 默 认 情 况 下 为 不 选中 。 

(2) 9.3.php 文件 中 ， 使 用 的 选项 name 值 为 achecked、bchecked、cchecked 并 根据 value 
值 做 出 判断 。 


9.3.4 ” 单 选 按钮 


下 面 通 过 案例 来 介绍 如 何 使 用 单 选 按钮 。 有 具体 操作 步骤 如 下 。 
ED 创建 文件 9.4.html。 代 码 如 下 : 


<html> 

<head> 

</head> 

<body> 

<form action="9.4.php" method="post"> 

<h3> 单 选 一 项 : </h3> 

<input type="radio" name="aradio" value="al"” /> 蓝天 
<input type="radio" name="aradio" value="a2" checked="checked" /> 和 白云 
<input type="radio" name="aradio" value="a3" /> 大 海 
<h3> 输 入 名 称 : </h3> 

<input type="text" name="name" size="10" /> 

</form> 

</body> 

</html> 


创建 文件 9.4.php。 代 码 如 下 : 


<?php 

$name = $_POST['name']7 
$aradio = $_POST['aradio']; 
if($aradio == "al’'){ 

echo "蓝天 "."<br/>"; 

}else if($aradio == 'a2'){ 
echo "白云 "."<br/>"; 

}elsef 

echo "大 海 "."<br/>"; 

} 

echo $name."<br/>"; 

及 多 


EES) 运行 9.4html， 结 果 如 图 9-7 所 示 。 选 中 单 选 按钮 并 在 文本 框 中 输入 内 容 后 ， 按 
Enter 键 确认 ， 结 果 如 图 9-8 所 示 。 
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(1) <input type="radio” name="aradio”value="al" 人 语句 定义 了 一 个 单 选 按钮 。 后 面 的 
<input type="radio" name="aradio" value="a2" checked="checked" /> 和 <input type="radio" 
name="aradio" value="a3" /> 定义 了 另外 的 两 个 单 选 按 钮 。 

定义 一 个 input 标识 为 单 选 按钮 时 需要 指定 类 型 为 radio: 

<input type="radio" ... /> 


定义 为 单 选 按钮 后 ， 还 需要 定义 单 选 按钮 的 name 属性 ， 以 确定 在 服务 器 端 程序 的 唯一 
性 。 定 义 value 属性 ， 以 确定 此 单 选 按钮 所 要 传递 的 值 。 定 义 checked 属性 ， 以 确定 单 选 按钮 
的 默认 状态 ， 若 为 checked 则 是 默认 的 选择 ， 如 果 不 定义 此 项 ， 默 认 情 况 下 为 不 选择 。 

(2) 在 9.4.php 文件 中 ， 使 用 了 单 选 按钮 的 name 值 为 “aradio”。 然 后 让 语句 通过 对 
aradio 传递 的 不 同 值 做 出 判断 ， 打 印 不 同 的 值 。 


9.3.5 下拉 列表 


下 面 通 过 案例 来 介绍 下 拉 列 表 的 使 用 方法 和 技巧 。 有 具体 操作 步骤 如 下 。 
EEDogp 创建 文件 9.5.html。 代 码 如 下 : 


<html> 

<head></head> 

<body> 

<form action="9.5.php" method="post"> 

<h3> 在 下 拉 菜 单 中 选择 一 项 : </h3> 

<select name="aselect" Size="1"> 

<option value="shanghai"> 上 海 </option> 
<option value="qingdao" selected> 青 岛 </option> 
<option value="beijing"> 北 京 </option> 
<option value="shenzhen"> 深 圳 </option> 
</select> 

<h3> 输 入 名 称 : </h3> 

<input type="text" name="name" size="10" /> 
</form> 

</body> 

</html> 


创建 文件 9.5.php。 代 码 如 下 : 


[ 口 
长 
第 
= 一 一 一 二 = ao 
ek: - OO x 诗 | 
Or @aEEsremaea:ols 本 
选项 | 文人 纺 和 (E) 二 看 (V) 收 基 夫 A) 工具 m ” 和 
DO 蓝天 口 自 云 @ 大 海 人 明 
输入 名 称 : 态 
| 效 
R125% ~ 到 125% ~ | 
图 9-7 使 用 单 选 按钮 图 9-8 程序 运行 结果 天 
与 
【案例 剖析 】 三 
8 
页 入 
面 
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<?php 
Sname = $_POST['name']; 
$aselect = $ POsST['aselect']; 
if($aselect == 'shanghai'){ 
echo "上海"."<bE />"; 
}else if(S$aselect == 'qingdao'){ 
echo "青岛 "."<br />"; 
}else if($aselect == 'beijing'){ 
echo "北京 "- "<br />"; 
}else{ 
echo "深圳 "."<br />"; 
i 
echo $name; 
区 全 2 





国 JE》 运行 9.5.html， 结 果 如 图 9-9 所 示 。 选 择 下 拉 菜 单 中 的 一 项 并 在 文本 框 中 输入 内 
容 后 ， 按 Enter 键 确 认 ， 结 果 如 图 9-10 所 示 。 




















OO -ol < 
EA 
在 下 拉 菜 单 中 选择 一 项 : ON, 
到 口 x 
上 @ -|@ pyocaho.. P ~ ©||& 
输入 名 称 : 文件 (久久 (E) 下 看 收藏 天 (A】 工具 1 ” 
上 海 
er] 最 爱 的 城市 
雪 125%6 > 乳 125% 
图 9-9 使 用 下 拉 列 表 图 9-10 程序 运行 结果 
【案例 剖析 】 


(1) 下 拉 列 表 是 通过 <select></select> 标 识 表 示 的 。 而 下 拉 列 表 当 中 的 选项 是 通过 包含 在 
其 中 的 <option></option> 标 识 表 示 的 。<select> 标 识 中 的 name=" 定 义 下 拉 列 表 的 name 属性 ， 
以 确认 它 的 唯一 性 。<option> 标 识 中 的 value="" 定 义 需要 传递 的 值 。 

(2) 在 9.5.php 文件 中 ， 则 使 用 了 选项 的 name 值 为 “aselect”。 然 后 让 语 句 通过 对 aselect 
传递 的 不 同 的 值 做 出 判断 ， 打 印 不 同 的 值 。 


9.3.6 ” 重 置 按钮 和 提交 按钮 


重 置 按钮 用 来 重 置 所 有 的 表单 输入 数据 。 

在 前 面 的 例子 中 ，form 里 的 所 有 元 素 都 已 经 设置 完成 ， 并 且 在 相应 的 PHP 文件 中 做 出 了 
处 理 。 这 个 时 候 ， 想 要 把 HTML 页 面 中 所 有 的 数据 发 送出 去 给 相应 PHP 文件 进行 处 理 ， 就 需 
要 使 用 submit 按钮 ， 也 就 是 提交 按钮 。 完 成 添加 提交 按钮 ， 并 且 提 交 数 据 。 

下 面 通过 一 个 综合 案例 来 学 习 重 置 按钮 和 提交 按钮 。 

具体 操作 步骤 如 下 。 

创建 文件 9.6.html 文件 。 代 码 如 下 : 


Gi 


<html> 

<head></head> 

<body> 

<form action="9.6.php" method="post"> 
<h3> 输 入 一 个 名 称 : </h3> 

<input type="text" name="name" size="10" /> 

<h3> 确 认 此 项 (可 复 选 ) : </h3> 

<input type="checkbox" name="achecked" checked="checked" value="1"” /> 
选择 此 项 传递 的 A 项 的 value 值 。 

<input type="checkbox" name="bchecked" value="2" /> 
选择 此 项 传递 的 B 项 的 value 值 。 

<input type="checkbox" name="cchecked" value="3" /> 
选择 此 项 传递 的 c 项 的 value 值 。 

<h3> 单 选 一 项 : </h3> 

<input type="radio" name="aradio" value="al" /> 蓝天 
<input type="radio" name="aradio" value="a2" checked="checked" /> 和 白云 
<input type="radio" name="aradio" value="a3" /> 大 海 
<h3> 在 下 拉 菜 单 中 选择 一 项 : </h3> 

<select name="aselect" size="1"> 

<option value="shanghai"> 上 海 </option> 

<option value="qingdao" selected> 青 岛 </option> 
<option value="beijing"> 北 京 </option> 

<option value="shenzhen"> 深 圳 </option> 

</select> 

<h3> 重 置信 息 : </h3> 

<input type="RESET" value=" 重 置 " /> 

<h3> 提 交 信 息 到 9. 6.php 文件 : </h3> 

<input type="submit"” value=" 提 交 " /> 

</form> 

</body> 

</html> 


创建 文件 9.6.php。 代 码 如 下 : 


<?php 

$name = $_POST['name']; 

if(isset($ POST['achecked'])){ 

$achecked = $_POST['achecked']7 

} 

if(isset($ POST['bchecked'])){ 

$bchecked = $_POST['bchecked']; 

} 

if(isset($ POST['cchecked'])){ 

$cchecked = $_POST['cchecked']; 

$aradio = $ POST['aradio']; 

$aselect = $_POST['aselect']7 

echo $name."<br />"; 

if(isset($achecked) and $achecked == 1)1{ 
echo "选项 A 的 value 值 已 经 被 正确 传递 。<br />"; 
}elsef 

echo "选项 A 没有 被 选择 ， 其 value 值 没有 被 传递 。<br />"; 
} 

if(isset(S$bchecked) and $bchecked == 2){ 
echo "选项 B 的 value 值 已 经 被 正确 传递 。<br />"; 
}elsef{ 
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echo "选项 B 没有 被 选择 ， 其 value 值 没有 被 传递 。<br />"; 
， 

if(isset($cchecked) and $cchecked == 3){ 

echo "选项 c 的 value 值 已 经 被 正确 传递 。<br />"; 

jelsef 

echo "选项 c 没有 被 选择 ， 其 value 值 没有 被 传递 。<br />"; 
} 

4f($aradio ==" "al){ 

echo "蓝天 <br />"; 

}else if($aradio == 'a2'){ 

echo "白云 <br />"; 

}else{ 

echo "大 海 <br />"; 

1 

if($aselect == ' 上 海 '){ 

echo "上 海 <br/>"7 





J}else if($aselect == 'gqingdao'){ 
echo "青岛 <br/>"; 

}else if($aselect == 'beijing'){ 
echo "北京 <br />"; 

}elsel{ 


echo "深圳 "; 
1 

2 

</BODY> 
</HTML> 


SS 运行 9.6.html， 结 果 如 图 9-11 所 示 。 
EC 单 击 “ 提 交 ” 按 钮 ， 页 面 将 会 跳 转 到 9.6.php， 输 出 结果 如 图 9-12 所 示 。 
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| 
| 回 选择 此 项 传道 的 A 项 的 value 值 。 思 选择 此 项 传递 的 B 项 的 value 值 . 
口 选择 此 项 传递 的 5 项 的 Yalue 值 





























单 选 一 项 。 
OO 阳 - OO x 
ee @ [@-G pocaho. DP - © | Biocalhost 
在 下 拉 菜 单 中 选择 一 项 : 文件 (月 ”篇 名 (E) 。 音 看 (V) 收藏 奖 (A) ”工具 (T) 帮助 (H) 
到 综合 测试 
重 置信 息 : 选项 A 的 value 值 已 经 被 正确 传递 . 
i 选项 B 的 value 值 已 经 被 正确 传递。 
LE 选项 C 没 有 被 选择 ， 其 value 值 没有 被 传递 。 
提交 信息 到 9. 6. php 文 件 : Hs 
家 
所 六 | 
| 于 A 或 125% > 
图 9-11 综合 应 用 效果 图 9-12 单 击 “提交 ”按钮 后 的 结果 
【案例 剖析 】 


由 <input type="RESET" value=" 重 置 "> 语 句 可 见 ， 重 置 按钮 是 <input 亿 标 识 的 一 种 。 定 义 


一 个 input 标识 为 单 选 项 的 必要 因素 为 : 
<input tyYpe="reset"”-.。/> 


Value 属性 是 按钮 所 显示 的 字符 。 


9.4 传递 数据 
数据 传递 的 常用 方法 为 POST 和 GET 两 种 。 下 面 介 绍 这 两 种 方法 的 使 用 技巧 。 


9.4.1 用 POST 方式 传递 数据 


表单 传递 数据 是 通过 POST 和 GET 两 种 方式 进行 的 。 在 定义 表单 属性 的 时 候 ， 都 要 在 
method 属性 上 定义 使 用 哪 种 数据 传递 方式 。 

<form action="URI" method="post"> 定 义 了 此 表单 在 把 数据 传递 给 目标 文件 时 使 用 的 是 
POST 方式 。<form action="URI" method="get"> 则 定义 了 此 表单 在 把 数据 传递 给 目标 文件 的 时 
候 ， 使 用 的 是 GET 方式 。 

POST 方式 是 比较 常见 的 表单 提交 方式 。 通 过 POST 方式 提交 的 变量 ， 不 受 特定 变量 大 小 
的 限制 ， 并 且 被 传递 的 变量 不 会 在 浏览 器 地 址 栏 里 以 URL 的 方式 显示 出 来 。 

上 一 节 中 的 案例 都 是 采用 POST 方式 传递 数据 ， 这 里 就 不 再 举例 说 明 。 


9.4.2 用 GET 方式 传递 数据 


GET 方式 比较 有 特点 。 通 过 GET 方式 提交 的 变量 ， 有 大 小 限制 ， 不 能 超过 100 个 字符 。 
它 的 变量 名 和 与 之 相对 应 的 变量 值 都 会 以 URL 的 方式 显示 在 浏览 器 地 址 栏 里 。 因 此 ， 若 传递 
大 而 敏感 的 数据 ， 一 般 不 使 用 此 方式 。 

使 用 GET 方式 传递 数据 ， 通 常 是 借助 于 URL 进行 的 。 

下 面 通 过 案例 对 此 操作 进行 讲解 。 具 体操 作 步 骤 如 下 。 

ED 创建 文件 9.7.php。 代 码 如 下 : 


<?php 
if(!$_GET['u']) 


echo ' 参 数 还 没有 输入 。'; 
J}elsef{ 

$user = $ GET['u']7 
switch ($user){ 

case 1: 

echo "您 最 喜欢 的 是 苹果 "; 


break; 


加 六 到 别 qaM JT dHd 一 一 站 涤 讲 昌 骂 慢 订 翰 6 洲 加 





case 22 
echo "您 最 喜欢 的 是 香蕉 "; 
break; 
case 3: 


echo "您 最 喜欢 的 是 哈密 瓜 "; 
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break; 
} 


二 


EDP》 在 浏览 器 的 地 址 栏 中 输入 “http:Wlocalhostycode/ch09/9.7.php?u”， 并 按 Enter 键 
确认 ， 运 行 结果 如 图 9-13 所 示 。 
ERe) 在 浏览 器 的 地 址 栏 中 输入 “http://localhost/getparam.php?u=1”， 并 按 Enter 键 确 
认 ， 运 行 结果 如 图 9-14 所 示 。 
= 日 x 
[@-|@ hpiocaho- P -| 全 
文件 (六 纺 (E) 二 看 VI 收 夫 A 工具 m ” 
参数 还 没有 输入 。 








一 口 x 
@ [@- Opecaho. P- © || Glocs 
文件 由 篇 坟 (E) 可 看 (V) 收 大 天 A) 工具 m 规 ” 
您 最 喜欢 的 是 苹果 


























矶 125% ~ 


图 9-13 ”程序 运行 结果 一 





R125% ~ 
图 9-14 程序 运行 结果 二 


EYY 在 浏览 器 的 地 址 栏 中 输入 “http://localhost/getparam.php?u=3”， 并 按 Enter 键 确 
认 ， 运 行 结果 如 图 9-15 所 示 。 





- DO x 
OB htpi//localho.. P ~ © | @ local 
文件 月 。 六 总 (E) 下 看 (V) 收藏 (A) 工具 TT) 邦 |” 


您 最 喜欢 的 是 哈密 瓜 








戈 125% ~ 
图 9-15 “程序 运行 结果 三 
【案例 剖析 】 
(1) 在 URL 中 ，GET 方式 通过 “?” 号 后 面 的 数组 元 素 的 键 名 (这 里 是 “u”) 来 获得 元 素 
的 值 。 
(2) 对 元 素 赋值 使 用 “=” 号 。 
(3) switch 条 件 语句 做 出 判断 ， 并 返回 结果 。 


9.5 ”PHP 获取 表单 传递 数据 的 方法 


如 果 表 单 使 用 POST 方式 传递 数据 ， 则 PHP 要 使 用 全 局 变量 数组 $ POST[] 来 读 取 所 传递 
的 数据 。 


在 表单 中 ， 元 素 传 递 数 据 给 $_ POST 全 局 变量 数组 ， 其 数据 以 关联 数组 中 的 数组 元 素 形 
式 存在 。 以 表单 元 素 的 名 称 属性 为 键 名 ， 以 表单 元 素 的 输入 数据 或 是 传递 的 数据 为 键 值 。 
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例如 ，9.6.php 文件 中 的 Sname=$ POST[mame'] 语 句 就 是 读 取 名 为 name 的 文本 框 中 的 数 
据 。 此 数据 是 以 name 为 键 名 ， 以 文本 框 输入 的 数据 为 键 值 。 

再 如 ，$achecked=-$_ POST['achecked'] 语 句 读 取 名 为 achecked 的 复 选 框 传递 的 数据 。 此 数 
据 是 以 achecked 为 键 名 ， 以 复 选 框 传递 的 数据 为 键 值 。 

如 果 表 单 使 用 GET 方式 传递 数据 ， 则 PHP 要 使 用 全 局 变量 数组 $ GET[] 来 读 取 所 传递 的 
数据 。 与 $ POST[] 相 同 ， 表 单 中 元 素 传递 数据 给 8$_GET[] 全 局 变量 数组 ， 其 数据 以 关联 数组 中 
的 数组 元 素 形式 存在 。 以 表单 元 素 的 名 称 属 性 为 键 名 ， 以 表单 元 素 的 输入 数据 或 是 传递 的 数 
据 为 键 值 。 


9.6 PHP 对 URL 传递 的 参数 进行 编码 


PHP 对 URL 中 传递 的 参数 进行 编码 ， 一 是 可 以 实现 对 所 传递 数据 的 加 密 : 二 是 可 以 对 无 
法 通过 浏览 器 进行 传递 的 字符 进行 传递 。 

实现 此 操作 一 般 使 用 urlencode0 函 数 和 rawurlencode0) 函 数 。 而 对 此 过 程 的 反 向 操作 就 是 
使 用 urldecode0 函 数 和 rawurldecode() 函 数 。 

下 面 通过 案例 对 此 操作 进行 讲解 。 具 体操 作 步 又 如 下 。 

创建 文件 9.8.php 文件 。 代 码 如 下 : 


<?php 

$user = ' 王 小 明 刘 晓 莉 ' ; 

$linkl = "index.php?userid=".urlencode ($user)."<br/>"; 
$link2 = "index.php?userid=".rawurlencode ($user) ."<br/>"7 
echo $1linkl.$1link2; 

echo urldecode ($1ink1); 

echo urldecode ($1ink2); 

echo rawurldecode ($1ink2) 7 

?> 


ED 程序 运行 结果 如 图 9-16 所 示 。 
| 二 ”如 “ 贡 
[加 -|@ Mp/focahosveocd Pp - © | Bocalhost 全 7 


文件 月 。 渴 号 日” 豆 看 收 辣 夫 (A) 工具 (大 用 (1 


index. php2userid-%CD%F5%DONA1YWC3%F7+XC1%F5%CF%FEXCO%F2 
index. php?userid=XCDAF5%DOXAI1%C3%F7%20XC1%F5%CFXFEWCO%F2 
index. php?userid= 王 小 明 刘 晓 莉 
index. php?userid= 王 小 明 刘 晓 莉 
index. php?userid= 王 小 明 刘 晓 莉 
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9-16 ”程序 运行 结果 
【案例 剖析 】 


(1) 在 $linkl 变量 的 赋值 中 ， 使 用 urlencode0 函 数 对 一 个 中 文字 符 串 $user 进行 编码 。 
(2) 在 $link2 变量 的 赋值 中 ， 使 用 rawurlencode() 函 数 对 一 个 中 文字 符 串 $user 进行 编码 。 
(3) 这 两 种 编码 方式 的 区 别 在 于 对 空格 的 处 理 ，urlencode0 函 数 将 空格 编码 为 “+” 号 ， 
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而 rawurlencode0 〇 函数 将 空格 编码 为 “%20”。 
(4) urldecode0 函 数 实现 对 编码 的 反 向 操作 。 


9.7 ”案例 实战 一 一 团购 商品 订单 表 


下 面 进行 处 理 表单 数据 的 讲解 。 此 案例 中 ， 将 模拟 团购 商品 订单 表 。 有 具体 操作 步骤 如 下 。 
EC 创建 文件 9.9.html。 代 码 如 下 : 


<html> 

<head> 

<h2> 酷 客 团购 商品 订单 表 </h2> 

</head> 

<body> 

<form action="9.9.php" method="post"> 

<table> 

<tr bgcolor="#3399FF"> 

<td> 客 人 姓名 :</td> 

<td><input type="text" name="customername" size="10" /></td> 
</Er> 

<tr bgcolor="#CCCCCC"> 

<td> 商 品名 称 </tq> 

<td><input type="text" name="cname" size="8" /></td> 

本 多 在 

<tr bgcolor="#CCCCCC"> 

<td> 采 购 数 目 </td> 

<td><input type="text" name="counts" size="8" /> 台 </td> 
</tr> 

<tr bgcolor="#3399FF"> 

<tq> 联 系 电话 :</td> 

<td><input type="text" name="phone" size="15" /></td> 
/EF 

<tr bgcolor="#666666"> 

<td align="center"><input type="submit" value=" 确 认 团购 信息 " /></td> 
</Er> 

</table> 

</form> 

</body> 

</html> 


创建 一 个 PHP 文件 9.9.php。 代 码 如 下 : 


<?php 

$customername = $_POST['customername']; 

$cname= $_POST['cname']; 

$counts = $_POST['counts']; 

$phone = $_POST['phone']; 

echo '<p> 确 认 采 购 信息 :</p>"'; 

echo ' 尊 敬 的 客户 ' .$customername .'! 您 采购 的 '， .$cname. ' 数 量 为 ' .$counts.'， 您 的 联系 
电话 是 '.$phone.'。'; 


Et 


区 了 TE》 运行 文件 9.9.html， 结 果 如 图 9-17 所 示 。 














OT Er 5-0 Sean 
文件 (F) ”六 霹 (E) ”可 看 (V) 家 写 夫 (入 。 工 其 (T) 帮助 (H) 


酷 客 团购 商品 订单 表 


| lem | 
| 


采购 数目 135 和 台 
| | 
确认 团购 信息 
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图 9-17 9.9.html 的 运行 结果 
输入 信息 后 ， 单 击 “ 确 认 团购 信息 ”按钮 ， 结 果 如 图 9-18 所 示 。 





- DO x 





国 -全 hp//ocahoscode D ©|| Blocalhost 
文件 (月 浪 同 (E) ”至 看 (V) 收藏 灾 (A) 工具 (T) 帮 盈 (H) 


确认 采购 信息 : 
尊敬 的 客户 王 飞 肯 ! 您 采购 的 海尔 冰箱 数量 为 135， 您 的 联系 电话 是 18012345678。 








扰 125% 
图 9-18 9.9.php 的 运行 结果 
【 娄 例 剖析 】 
(1) 在 9.9.html 中 的 form 通过 POST 方法 (method) 把 3 个 <input type="text" .… /> 中 的 文本 
数据 发 送 给 9.9.php。 


(2) 在 9.9.php 中 ， 代 码 读 取 数 组 $ POST 中 的 具体 变量 $ POST['customername']、$_POST 
[cname]、$_POST['counts]、$_POST[phone]， 并 赋值 给 本 地 变量 $customername、$cname、 
$counts、$phone。 然 后 ， 通 过 echo 命令 使 用 本 地 变量 ， 把 信息 生成 HTML 后 输出 给 浏览 器 。 

(3) 要 提 到 的 是 “echo ' 尊 散 的 客户 '.$Scustomername.'! 您 采购 的 '$cname:' 数 量 为 
'.$counts.'， 您 的 联系 电话 是 .$phone'。':” 中 的 “.” 是 字符 串 连 接 操 作 符 。 它 把 不 同 部 分 的 
字符 串 连 接 在 一 起 。 在 使 用 echo 命令 的 时 候 经 常会 用 到 。 


9.8 疑难 解 惑 


疑问 1: 使 用 urlencode() 和 rawurlencode() 函 数 需 要 注意 什么 ? 

答 : 要 注意 的 是 ， 如 果 配 合 JS 处 理 页 面 信息 的 话 ， 要 注意 使 用 urlencode0 函 数 后 “+” 
号 与 JS 的 冲突 。 由 于 JS 中 “+” 号 是 字符 串 类 型 的 连接 操作 符 才 处 理 ， 和 否则 URL 就 无 法 识 
别 其 中 的 “+” 号 。 这 时 ， 可 以 使 用 rawurlencodeO 函 数 对 其 进行 处 理 。 
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疑问 2: GET 和 了 POST 的 区 别 和 联系 是 什么 ? 


答 : 二 者 的 区 别 与 联系 如 下 。 
(1) POST 是 向 服务 器 传送 数据 ;GET 是 从 服务 器 上 获取 数据 。 
(2) POST 是 通过 HTTP POST 机 制 将 表单 内 各 个 字段 及 其 内 容 放 置 在 HTML HEADER 
内 一 起 传送 到 ACTION 属性 所 指 的 URL 地 址 ， 用 户 看 不 到 这 个 过 程 。GET 是 把 参数 数据 队 
列 加 到 提交 表单 的 ACTION 属性 所 指 的 URL 中 ， 值 和 表单 内 各 个 字段 一 一 对 应 ， 在 URL 中 
可 以 看 到 。 
(3) 对 于 GET 方式 ， 服 务 器 端 用 Request.QueryString 获取 变量 的 值 ， 对 于 POST 方式 ， 
服务 器 端 用 Request.Form 获取 提交 的 数据 。 
: (4) POST 传送 的 数据 量 较 大 ， 一 般 默认 为 不 受 限制 。 
(5) POST 安全 性 较 高 ，GET 安全 性 非常 低 ， 但 是 执行 效率 却 比 POST 方法 高 。 
(6) 在 做 数据 添加 、 修 改 或 删除 时 ， 建 议 用 POST 方式 ; 而 在 做 数据 查询 时 ， 建 议 用 
GET 方式 。 
(7) 对 于 机 密 信息 的 数据 ， 建 议 采 用 POST 数据 提交 方式 。 
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第 10 音 
时 间 很 重要 一 一 
管理 日 期 和 时 间 


日 期 和 时 间 对 于 很 多 应 用 来 说 是 十 分 敏感 的 。 程 序 中 ， 很 多 情况 下 都 是 依靠 日 
期 和 时 间 才 能 做 出 判断 、 完 成 操作 的 。 例 如 ， 酒 店 商务 网 站 中 查看 最 新 的 房价 情 
况 ， 这 与 时 间 是 密 不 可 分 的 。 本 章 介绍 日 期 和 时 间 的 获得 及 格式 化 方面 的 内 容 。 
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10.1 系统 时 区 的 设置 


这 里 的 系统 时 区 是 指 运行 PHP 的 系统 环境 。 常见 的 有 Windows 系统 和 UNIX-like( 类 
UNIX) 系 统 。 对 于 它们 时 区 的 设置 ， 关 系 到 运行 应 用 的 时 间 准 确 性 。 


10.1.1 时 区 划分 


时 区 划分 是 一 个 地 理 概念 。 从 本 初子 午 线 开 始 向 东 和 向 西 各 有 12 个 时 区 。 比 如 ， 我 们 的 
北京 时 间 是 东 八 区 ， 美国 太平 洋 时 间 是 西 八 区 。 在 Windows 系统 里 ， 这 个 操作 比较 简单 。 在 
控制 面板 里 设置 就 行 了 。 在 Linux 这 样 的 UNIX-like 系统 中 ， 需 要 使 用 命令 对 时 区 进行 设置 。 


10.1.2 ”时 区 设置 


在 PHP 中 ， 日 期 时 间 的 默认 设置 是 GMT 格林 尼 治 时 间 。 在 使 用 时 间 日 期 功能 之 前 ， 需 
要 对 时 区 进行 设置 。 

时 区 的 设置 方法 主要 有 以 下 两 种 。 

(1) 修改 php.ini 文件 的 设置 。 找 到 “:date.timezone=” 选 项 ， 将 其 值 修改 为 date.timezone 
=Asia/Hong Kong， 这 样 系统 默认 时 间 为 东 八 区 的 时 间 。 

(2) 在 应 用 程序 中 直接 用 函数 date_default_ timezone _set0) 来 设置 。 其 语法 格式 如 下 : 


date default timezone set ("timezone") 


参数 timezone 为 PHP 可 以 识别 的 时 区 名 称 。 例 如 ， 设 置 我 国 北京 时 间 可 以 使 用 的 时 区 包 
括 : PRC( 中 华人 民 共 和 国 )、Asia/Chongqing( 重 庆 )、Asia/Hong Kong (香港 )、Asia/ 
Shanghai( 上 海 ) 等 。 这 些 时 区 的 名 称 都 是 有 效 的 。 

这 种 方法 设置 时 比较 灵活 。 设 置 完成 后 ，data0 函 数 便 可 以 正常 使 用 ， 不 会 再 出 现时 差 问题 。 





10.2 PHP 的 日 期 和 时 间 函 数 
下 面 开始 学 习 PHP 的 常用 日 期 和 时 间 函 数 的 使 用 方法 和 技巧 。 
10.2.1 关于 UNIX 时 间 戳 


在 很 多 情况 下 ， 程 序 需要 对 日 期 进行 比较 、 运 算 等 操作 。 如 果 按 照 人 们 日 常 的 计算 方 
法 ， 很 容易 知道 6 月 5 号 和 6 月 8 号 相差 几 天 。 

然而 ， 如 果 日 期 的 书写 方式 是 2012-3-8 或 2012 年 3 月 8 日 星期 五 ， 这 让 程序 如 何 运算 
呢 ? 对 整 型 数据 的 数学 运算 来 说 ， 好 像 这 样 的 描述 并 不 容易 处 理 。 如 果 想 知道 3 月 8 号 和 4 
月 23 号 相差 几 天 ， 则 需要 把 月 先 转换 为 30 天 或 31 天 ， 再 对 剩余 天 数 加 减 。 这 是 一 个 很 麻烦 
的 过 程 。 


如 果 时 间或 者 日 期 是 一 个 连贯 的 整数 ， 这 样 处 理 起 来 就 很 方便 了 。 

幸运 的 是 ， 系 统 的 时 间 正 是 以 这 种 方式 储存 的 ， 这 种 方式 就 是 时 间 戳 ， 也 称 为 UNIX 时 
间 玲 。UNIX 系统 和 UNIX-like 系统 把 当下 的 时 间 储 存 为 32 位 的 整数 ， 这 个 整数 的 单位 是 
秒 ， 而 这 个 整数 的 开始 时 间 为 格林 尼 治 时 间 (GMT) 的 1970 年 1 月 1 日 的 零点 整 。 换 句 话 说 ， 
就 是 现在 的 时 间 是 GMT 1970 年 1 月 1 日 的 零点 整 到 现在 的 秒 数 。 

由 于 每 一 秒 的 时 间 都 是 确定 的 ， 这 个 整数 就 像 一 个 章 戳 一 样 不 可 改变 ， 所 以 就 称 为 UNIX 
时 间 戳 。 

这 个 时 间 惟 在 Windows 系统 下 也 是 成 立 的 ， 但 是 与 UNIX 系统 不 同 的 是 ，Windows 系统 
下 的 时 间 戳 只 能 为 正 整 数 ， 不 能 为 负 值 。 所 以 想 用 时 间 惟 表示 1970 年 1 月 1 日 以 前 的 时 间 是 
不 行 的 。 

PHP 则 是 完全 采用 了 UNIX 时 间 戳 的 。 所 以 不 管 PHP 在 哪个 系统 下 运行 ， 都 可 以 使 用 
UNIX 时 间 戳 。 


10.2.2 ”获取 当前 的 时 间 戳 NN 
要 获得 当前 时 间 的 UNIX 时 间 蕉 ， 以 用 于 得 到 当前 时 间 ， 直 接 使 用 time0 函 数 即 可 。 NN 
time(O 函 数 不 需 要 任何 参数 ， 直 接 返回 当前 日 期 和 时 间 。 
【 例 10.1】 获 取 当 前 的 时 间 戳 (示例 文件 ch10\10.1.php)。 
<?php 
Stl = time(); 


echo "当前 时 间 惟 为 : ".$t1; 


D> 


ij 节 寺 并 卫 酋 磺 一 一 阐 师 谨 画 字 十 o 站 国 


豆 














程序 运行 结果 如 图 10-1 所 示 。 一 一 x 
【案例 剖析 】 et 
(D 在 图 10-1 中 ,数值 1501847269 表示 从 1970 年 1 月 生生 1501847269 


1 日 0 点 0 分 0 秒 到 本 程序 执行 时 间 隐 的 秒 数 。 ee 
(2) 如 果 每 隔 一 段 时 间 刷 新 一 次 页 面 ， 获 取 的 时 间 蕉 的 

值 将 会 增加 。 这 个 数 会 一 直 不 断 地 变 大 ， 即 每 过 1 秒 ， 此 值 ”图 10 获取 当前 的 时 间 玫 

就 会 加 1。 


10.2.3 ”获取 当前 的 日 期 和 时 间 


可 使 用 date0 函 数 返回 当前 日 期 。 如 果 在 date0 函 数 中 使 用 参数 “U”， 则 可 返回 当前 时 
间 的 UNIX 时 间 戳 。 如 果 使 用 参数 “d”， 则 可 直接 返回 当前 月 份 的 01 到 31 号 的 两 位 数 日 
期 ， 等 等 。 

dateO 函 数 有 很 多 参数 ， 具 体 含义 如 表 10-1 所 示 。 











. 
14 傅 


PHP+MySQL 动态 网 站 开发 
案例 课堂 (第 2 版 ) $… 


表 10-1 date() 函 数 的 参数 




















参 数 含义 参 数 含义 
a 小 写 am 或 pm A 大 写 AM 或 PM 
d 01 到 31 的 日 期 D Mon 到 Sun 的 简写 星期 
e 显示 时 区 
月 份 的 全 拼 单词 
g 12 小 时 格式 的 小 时 数 (1 到 12) G 24 小 时 格式 的 小 时 数 (0 到 23) 
h | 12 小 时 格式 的 小 时 数 (01 到 12) | #24 小 时 格式 的 小 时 数 (00 到 23) 
i 分 钟 数 (01 到 60) | 1 | Daylight 
j 一 月 中 的 天 数 (从 1 到 31) 


1 | 一 周 中 天 数 的 全 拼 ee 
| mw | 


m 月 份 (从 01 到 12) 3 个 字母 的 月 份 简 写 ( 从 Jan 到 Dec) 
n 月 份 (从 1 到 12) 
| o@ | 与 格林 尼 治 时 间 相差 的 时 间 
s “| 利 数 (从 0o0 到 59) | s | 天 数 的 序数 表达 (st、nd、rd、th) 
1 | 一 个 月 中 天 数 的 总 数 (从 28 到 31) 时 区 简写 
当前 的 UNIX 时 间 巩 
Ww 数字 表示 的 周 天 ( 从 0-Sunday 到 6-Saturday) ISO8601 标准 的 一 年 中 的 周 数 
四 位 数 的 公元 纪年 (从 1901 到 2038) 
以 秒表 现 的 时 区 (从 -43200 到 50400) 





2 一 年 中 的 天 数 (从 0 到 364) 


10.2.4 ”使 用 时 间 戳 获取 日 期 信息 

如 果 相 应 的 时 间 戳 已 经 储存 在 数据 库 中 ， 程 序 需 要 把 时 间 戳 转化 为 可 读 的 日 期 和 时 间 ， 
才能 满足 应 用 的 需要 。 

PHP 中 提供 了 date0 和 getdateO 等 函数 来 实现 从 时 间 戳 到 通用 时 间 的 转换 。 

1. dateO) 函 数 

date0 函 数 主要 是 将 一 个 UNIX 时 间 戳 转化 为 指定 的 时 间 /日 期 格式 。 该 函数 的 语法 格式 如 下 : 

string date(string format， [时 间 惟 整数 ] ) 

此 函数 将 会 返回 一 个 字符 串 。 该 字符 串 就 是 一 个 指定 格式 的 日 期 时 间 ， 其 中 format 是 一 
个 字符 串 ， 用 来 指定 输出 的 时 间 格 式 。 时 间 戳 整数 可 以 为 空 ， 如 果 为 空 ， 则 表示 为 当前 时 间 
的 UNIX 时 间 戳 。 

format 参数 是 由 指定 的 字符 构成 的 ， 具 体 字符 的 含义 如 表 10-2 所 示 。 
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表 10-2 format 字符 的 含义 




















format 字符 含义 
a am 或 pm 
A AM 或 PM 
d 几 日 ， 二 位 数字 ， 若 不 足 二 位 ， 则 前 面 补 零 。 例 如 01 至 31 
D 星期 几 ，3 个 英文 字母 。 例 如 Fri 
F 月 份 ， 英 文 全 名 。 例 如 January 
h 12 小 时 制 的 小 时 。 例 如 01 至 12 
H 24 小 时 制 的 小 时 。 例 如 00 至 23 
g 12 小 时 制 的 小 时 ， 不 足 二 位 不 补 零 。 例 如 1 至 12 
G 24 小 时 制 的 小 时 ， 不 足 二 位 不 补 零 。 例 如 0 至 23 
i 分 钟 。 例 如 00 至 59 
j 几 日 ， 二 位 数字 ， 若 不 足 二 位 不 补 零 。 例 如 1 至 31 
1 星期 几 ， 英 文 全 名 。 例 如 Friday 
m 月 份 ， 二 位 数字 ， 若 不 足 二 位 则 在 前 面 补 零 。 例 如 01 至 12 
n 月 份 ， 二 位 数字 ， 若 不 足 二 位 则 不 补 零 。 例 如 1 至 12 
M 月 份 ，3 个 英文 字母 。 例 如 Jan 
s 秒 。 例 如 00 至 59 
S 字 尾 加 英文 序数 ， 两 个 英文 字母 。 例 如 也 、nd 
t 指定 月 份 的 天 数 。 例 如 28 至 31 
U 总 秒 数 
WwW 数值 型 的 星期 几 。 例 如 0( 星 期 日 ) 至 6( 星 期 六 ) 
将 年 ， 四 位 数字 。 例 如 1999 
y 年 ， 二 位 数字 。 例 如 99 





z 一 年 中 的 第 几 天 。 例 如 0 至 365 


下 面 通过 一 个 例子 来 理解 format 字符 的 使 用 方法 。 
【 例 10.2】 使 用 format 字符 (示例 文件 ch10\10.2.php)。 


<?php 

date default timezone_ set ("PRC"); 

// 定 义 一 个 当前 时 间 的 变量 

$tt = time(); 

echo "目前 的 时 间 为 : <br>"; 

// 使 用 不 同 的 格式 化 字符 测试 输出 效果 

echo date("Y 年 m 月 d 日 [1]H 点 i 分 s 秒 ",$tt)."<br>"; 
echo date("y-m-d h:i:s a",$tt)."<br>"; 

eeho date("Y MD HoT SA" SLE) "<Dbro"s 
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echo date( rary Lott). <br>"p 
echo date("Y-M-D H:I:S",$tt)." <br>"; 











2> 
程序 运行 结果 如 图 10-2 所 示 。 
有 一 口 x 
@@-G he/ioalho. Pp - © | Blocaltost x 
文件 (有) 妨 强 E) 至 看 (V) 。 收 茂 夫 (A) ”工具 (] 。 帮助 (H) 
目前 的 时 间 为 : 
2017 年 08 月 04 日 [Friday]19 点 52 分 41 秒 
17-08-04 07:52:41 pm 
2017-Aug-Fri 19:0:th PM 
August, 04, 17 Friday 
2017-Aug-Fri 19:0:th 
Eo 
10-2 使 用 format 字 符 
【案例 训 析 】 


(1) date_default_ timezone_set("PRC") 语 句 的 作用 是 设置 默认 时 区 为 北京 时 间 。 如 果 不 设 
将 会 显示 安全 警告 信息 。 
(2) 格式 化 字符 的 使 用 方法 非常 灵活 ， 只 要 设置 字符 串 中 包含 的 字符 ，date() 函 数 就 能 将 


字符 串 蔡 换 成 指定 的 日 期 信息 。 利 用 上 面 的 函数 可 以 随意 输出 自己 需要 的 日 期 。 


由 


2. getdate() 函 数 


getdate() 函 数 可 以 获取 详细 的 时 间 信 息 。 该 函数 的 语法 格式 如 下 : 
array getdate (时 间 礁 整数 ) 


getdate() 函 数 返回 一 个 数组 ， 包 含 日 期 和 时 间 的 各 个 部 分 。 如 果 它 的 参数 时 间 惟 整数 为 
则 表示 直接 获取 当前 时 间 惟 。 下 面 举 例 说 明 此 函数 的 使 用 方法 和 技巧 。 
【 例 10.3】 使 用 getdate0 函 数 (示例 文件 ch10\10.3.php)。 


<?php 

date default timezone_ set ("PRC"); 

// 定 义 一 个 时 间 的 变量 

$tm ="2018-08-08 08:08:08"; 

echo "时 间 为 : ". $tm. "<br>"; 

// 将 格式 转化 为 UNIX 时 间 戳 

$tp = Strtotime ($tm); 

echo "此 时 间 的 Unix 时 间 惟 为 : ".$tp. "<br>"; 
$arl = getdate ($tp); 

echo "年 为 : ". $arl ["year™] ."<br>"; 


echo "月 为 : ". $arl["mon"] ."<br>"; 
echo "日 为 : ". $arl["mday"] ."<br>"; 
echo "点 为 : ". $arl["hours"] ."<br>"; 
echo "分 为 : ". S$arl["minutes"] ."<br>"; 
echo " 秒 为 : ". $arl["seconds"] ."<br>"; 
2> 


程序 运行 结果 如 图 10-3 所 示 。 








一 昌文 
© ES 6| Brocton E 
Es) bd 下 二 V) OA IAM my) 
时 间 为 : 2018-08-08 08:08:08 
此 时 间 的 Ie 到 时 间 本 为; 1533686888 

年 为 : 20 
月 为 : 8 
日 为 : 8 
点 为 : 8 
分 为 : 8 
秒 为 : 8 

成 125% = 





10-3 ”使 用 getdate() 函 数 


10.2.5 ”检验 日 期 的 有 效 性 


使 用 用 户 输入 的 时 间 数 据 时 ， 有 时 会 由 于 用 户 输入 的 数据 不 规范 ， 导 致 程序 运行 出 错 。 为 
了 检查 时 间 的 合法 有 效 性 ， 需 要 使 用 checkdate0 函 数 对 输入 日 期 进行 检测 。 它 的 语法 格式 如 下 : 
checkdate (月 份 ， 日 期 ， 年 份 ) 


此 函数 检查 的 项 目 是 : 年 份 整数 是 否 在 0~32767 之 间 ; 月 份 整数 是 否 在 1~12 之 间 ; 日 期 
整数 是 否 在 相应 的 月 份 的 天 数 内 。 下 面 通过 例子 来 讲述 如 何 检查 日 期 的 有 效 性 。 

【 例 10.4】 使 用 checkdate0) 函 数 (示例 文件 ch10\10.4.php)。 

<?php 

if (checkdate (2, 31, 2018) ){ 

echo "这 不 可 能 。 

}else{ 

echo "2 月 没有 31 号 。"; 

} 


>. 


程序 运行 结果 如 图 10-4 所 示 。 


ij 字 夫 涛 卫 并 夺 一 一 阐 贞 谨 画 字 二 or 当 国 


豆 





x 


一 口 
© -| hp//ocalho P ~ © | Blocalhost 


文件 介 。 坊 各 人 E) 豆 看 (V) 收藏 夫 (A) 工具 (T) 帮助 (H) 
2 月 没有 31 号 。 
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图 10-4 使 用 checkdate() 函 数 对 输入 日 期 进行 检测 


10.2.6 ”输出 格式 化 时 间 戳 的 日 期 和 时 间 


使 用 strftimeO 可 以 把 时 间 戳 格式 化 为 日 期 和 时 间 。 它 的 语法 格式 如 下 : 

strftime (格式 ， 时 间 戳 ) 

其 中 有 两 个 参数 ， 格 式 决 定 了 如 何 把 其 后 面 时 间 戳 格式 化 并 且 输 出 出 来 。 如 果 时 间 戳 为 
空 ， 则 系统 当前 时 间 戳 将 会 被 使 用 。 

格式 代码 的 含义 如 表 10-3 所 示 。 
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表 10-3 格式 代码 的 含义 


代码 含义 名义 











%a 周 日 期 ( 缩 简 ) %A 周 日 期 
%b 或 %h 月 份 ( 缩 简 ) %B 月 份 

vc 标准 格式 的 日 期 和 时 间 世纪 

%d 月 日 期 (从 01 到 31) %D 日 期 的 缩 简 格 式 (mmy/dd/yy) 

i 包含 两 个 字符 的 字符 串 月 日 期 

(从 01 到 31) 

%g 根据 周 数 的 年 份 (2 个 数字 ) %G 根据 周 数 的 年 份 (4 个 数字 ) 
%H 小 时 数 ( 从 00 到 23) 
%l 小 时 数 ( 从 1 到 12) 

%j 一 年 中 的 天 数 ( 从 001 到 366) 

%m 月 份 (从 01 到 12) %M 分 钟 (从 00 到 59) 


%n 新 一 行 ( 同 n) 

%p %P am 或 pm 

Yr 时 间 使 用 am 或 pm 表示 %R 时 间 使 用 24 小 时 制 表示 

%S 秒 ( 从 00 到 59) 

%t Tab( 同 \t) %T 时 间 使 用 hh:ss:mm 格式 表示 


一 年 中 的 周 数 (以 至 少 剩余 4 
天 的 这 一 周 开始 为 第 一 周 ) 








站 周 天 数 ee 一 年 中 的 周 数 ( 从 第 一 周 的 第 
- (从 1-Monday 到 7-Sunday) ” 一 个 星期 天 开始 ) 





Si 周 天 数 避 询 一 年 中 的 周 数 (从 第 一 周 的 第 
(从 0-Sunday 到 6-Saturday) 一 个 星期 一 开始 ) 

x 标准 格式 日 期 (无 时 间 ) %X 标准 格式 时 间 ( 无 日 期 ) 
%y 年 份 (2 字符 ) %Y 年 份 (4 字符 ) 

%z 和 %Z 时 区 

下 面 举例 介绍 用 法 。 
【 例 10.5】 输 出 格式 化 的 日 期 和 时 间 ( 示 例文 件 ch10\10.5.php)。 
<?php 


date default timezone set ("PRC"); 

echo (strftime ("%b %d %Y %X", mktime(20,0,0,12,31,98))); 

echo (gmstrftime ("%b %d %Y %X", mktime(20,0,0,12,31,98))); 

// 输 出 当前 日 期 、 时 间 和 时 区 

echo(gmstrftime("It is %a on %b %d, %Y, %xX time Zone: %2",time())); 
ye 


全 


程序 运行 结果 如 图 10-5 所 示 。 


| ¢ [OEE TE 
Se RH SEM WeseN TAT #09 





Dec 31 1998 20:00:00Dec 31 1998 12:00:00It is Mon on 
Aug 07，2017，11:25:56 time zone: 中 国标 准时 间 
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10-5 输出 格式 化 日 期 和 时 间 


10.2.7 显示 本 地 化 的 日 期 和 时 间 


由 于 世界 上 有 不 同 的 显示 习惯 和 规范 ， 所 以 日 期 和 时 间 也 会 根据 不 同 的 地 区 显示 为 不 同 
的 形式 。 这 就 是 日 期 时 间 的 本 地 化 显示 。 

实现 此 操作 需要 使 用 到 setlocale0 和 strftimeO 两 个 函数 。 后 者 已 经 介绍 过 。 

使 用 setlocale0 函 数 来 改变 PHP 的 本 地 化 默认 值 ， 实 现 本 地 化 的 设置 。 它 的 语法 格式 如 下 : 

setlocale (目录 ， 本 地 化 值 ) 


(1) 本 地 化 值 是 一 个 字符 串 ， 它 有 一 个 标准 格式 : language COUNTRY.chareacterset。 例 
如 ， 想 把 本 地 化 设 为 美国 ， 按 照 此 格式 为 en_US.utf8; 如 果 想 把 本 地 化 设 为 英国 ， 按 照 此 格 
式 为 en_GB.utf8; 如 果 想 把 本 地 化 设 为 中 国 ， 且 为 简体 中 文 ， 按 照 此 格式 为 zh_CN.gb2312， 
或 者 zh_CN.utf8。 

(2) 目录 是 指 6 个 不 同 的 本 地 化 目录 ， 如 表 10-4 所 示 。 
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表 10-4 本 地 化 目录 

目录 说 明 
LC.ATE 为 后 面 其 他 的 目录 设 定 本 地 化 规则 的 目录 
LC COLLATE 字符 串 对 比 目 录 
LC_CTYPE 字母 划 类 和 规则 
LC MONETARY 货币 表示 规则 
LC NUMERIC 数字 表示 规则 
LC TIME 日 期 和 时 间 表示 规则 





由 于 这 里 要 对 日 期 和 时 间 进 行 本 地 化 设置 ， 需 要 使 用 到 的 目录 是 LC_TIME。 下 面 通过 例 

子 对 日 期 时 间 本 地 化 进行 讲解 。 
【 例 10.6】 日 期 和 时 间 本 地 化 (示例 文件 ch10\10.6.php)。 

<?php 

date default timezone set ("PRC"); 

date default timezone_set ("Asia/Hong_Kong"); // 设 置 时 区 为 中 国 时 区 

setlocale (LC TIME, "zh_CN.gb2312"); // 设 置 时 间 的 本 地 化 显示 方法 

echo strftime("%z"); // 输 出 所 在 的 时 区 
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程序 运行 结果 如 图 10-6 所 示 。+0800 是 东 八 区 (中 国标 准 0 x 














时 间 )。 Or 
【案例 剖析 】 ee 
(1) date default timezone set("Asia/Hong Kong") 设 定时 2 
区 为 中 国 时 区 。 图 10-6 “日 期 和 时 间 本 地 化 
(2) setlocale0 设 置 时 间 的 本 地 化 显示 方式 为 简体 中 文 
方式 。 


(3) strftime("9%z") 返 回 所 在 时 区 ， 其 在 页 面 上 显示 为 简体 中 文 方式 。 


10.2.8 将 日 期 和 时 间 解 析 为 UNIX 时 间 戳 


通过 给 定 的 日 期 和 时 间 ， 使 用 mktime0 函 数 可 以 生成 相应 的 UNIX 时 间 蕉 。 它 的 语法 格 
式 如 下 : 

mktime (小 时 ， 分 钟 ， 秒 ， 月 份 ， 日 期 ， 年 份 ) 

把 时 间 和 日 期 的 部 分 输入 相应 位 置 的 参数 ， 即 可 得 到 相应 的 时 间 戳 。 下 面 的 例子 介绍 此 
函数 的 应 用 方法 和 技巧 。 

【 例 10.7】 使 用 mktime0 函 数 (示例 文件 ch10\10.7.php)。 


<?php 
$timestamp = mktime(0,0,0,3,31,2018); 


- 0O x 
OO hlocalho.. P ~ || @ locahost 
文件 站 ”光缆 [E) 盏 看 V) 收藏 灾 (A) 工具 Tm 如 助 (H) 








echo $timestamp; 

人 

程序 运行 结果 如 图 10-7 所 示 。 1 

其 中 mktime(0,0,0,3,31,2018) 使 用 的 时 间 是 2018 年 3 月 10-.7 使 用 mktime() 函 数 
31 号 0 点 整 。 


10.2.9 日 期 时 间 在 PHP 和 MySQL 数据 格式 之 间 的 转换 


日 期 和 时 间 在 MySQL 中 是 按照 ISO8601 格式 储存 的 。 这 种 格式 要 求 以 年 份 打头 ， 如 
2018-03-08 这 种 格式 。 从 MySQL 读 取 的 默认 格式 也 是 这 种 格式 。 对 于 这 种 格式 我 们 中 国人 是 
比较 熟悉 的 。 这 样 在 中 文 应 用 中 ， 几 乎 可 以 不 用 转换 ， 就 直接 使 用 了 。 

但 是 ， 在 西方 的 表达 方法 中 ， 经 常 把 年 份 放 在 月 份 和 日 期 的 后 面 ， 如 March 08, 2018。 所 
以 ， 在 接触 到 国际 的 特别 是 符合 英语 使 用 习惯 的 项 目 时 ， 需 要 对 ISO8601 格式 的 日 期 时 间 做 
合适 的 转换 。 

有 意思 的 是 ， 为 了 解决 这 个 英文 使 用 习惯 和 ISO8601 格式 冲突 的 问题 ，MySQL 提供 了 把 
英文 使 用 习惯 的 日 期 和 时 间 转 换 为 符合 ISO8601 标准 的 两 个 函数 ， 它 们 是 DATE_ FORMATO 和 
UNIX_TIMESTAMPO。 这 两 个 函数 在 SQL 语言 中 使 用 。 有 具体 用 法 将 在 介绍 MySQL 时 详 述 。 
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10.3 ”案例 实战 1 一 一 实现 倒计时 功能 


对 于 未 来 的 时 间 点 实现 倒计时 ， 其 实 就 是 使 用 现在 的 当下 时 间 惟 和 未 来 的 时 间 点 进行 比 
较 和 运算 。 
下 面 通过 案例 来 介绍 如 何 实现 倒计时 功能 。 
【 例 10.8】 实 现 倒 计时 (示例 文件 ch10\10.8.php)。 
<?php 
$timestampfuture = mktime (0,0,0,05,01,2018); 


$timestampnow = time(); 
$timecount = $timestampfuture - $timestampnow; 


王 浊 并 卫 酋 磺 一 一 阐 贞 谨 画 字 二 or 当 国 
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$days = round ($timecount/86400); 
echo "今天 是 ".date('Y F d') .” ,距离 2018 年 5 月 1 号 的 时 间 蕉 ， 还 有 ".$days." 天 。"; 




















?> NS 
程序 运行 结果 如 图 10.8 所 示 。 SS 
一 口 x 
人 [@- | rp//ocalhosyced, P ~ © | Blocalhost 
文件 (有 ) ”六 名 (E) ” 理 看 (V) 收藏 夫 (A) ”工具 (D) 帮助 (H) 
今天 是 2017 August 07 , 距离 2018 年 5 月 1 号 的 时 间 戳 ， 还 有 266 天 。 
扎 125% ~ 
图 10-8 ”实现 倒计时 
【案例 剖析 】 


(1) time0 不 带 任何 参数 ， 所 生成 的 时 间 是 当前 时 间 。 
(2) Stimecount 是 现在 的 时 间 戳 距离 未 来 时 间 点 的 时 间 戳 的 秒 数 。 
(3) round($timecount/86400)， 其 中 86400 为 一 天 的 秒 数 ，S$timecount/86400 得 到 天 数 ， 


round0) 函 数 取 约 数 ， 得 到 天 数 。 


10.4 ”案例 实战 2 一 一 比较 两 个 时 间 的 大 小 


对 比较 两 个 时 间 的 大 小 来 说 ， 如 果 通 过 一 定形 式 的 日 期 和 时 间 进 行 比 较 ， 或 者 不 同 格式 
的 日 期 和 时 间 进行 比较 ， 都 不 方便 。 最 方便 的 方法 是 把 所 有 格式 的 时 间 都 转换 为 时 间 惟 ， 然 
后 比较 时 间 截 的 大 小 。 
下 面 通过 例子 来 比较 两 个 时 间 的 大 小 。 
【 例 10.9】 比 较 两 个 时 间 的 大 小 (示例 文件 ch10\10.9.php)。 
<?php 
$timestampA = mktime (0,0,0,3,31,2018); 


$timestampB = mktime (0,0,0,1,31,2018); 
if($timestampA > $timestampB){ 
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echo "2018 年 三 月 的 时 间 惟 数值 大 于 2018 年 一 月 的 。 "> 
}elseif ($timestampA < $timestampB){ 

echo "2018 年 三 月 的 时 间 稚 数值 小 于 2018 年 一 月 的 。"; 
jelsef 

echo "两 个 时 间 相 同 。"; 

} 


区 2 


程序 运行 结果 如 图 10-9 所 示 。 





- DO x 
"| htpi/ocalho.. P - ©¢ | @ localhost 
文件 月 。 疙 各 (E) 得 看 (V) 收藏 夫 (A) 工具 (帮助 (H) 


2018 年 三 月 的 时 间 改 数值 大 于 2018 年 一 月 的 。 























大 125% > 


图 10-9 比较 两 个 时 间 的 大 小 





10.5 疑难 解 惑 


疑问 1: 如 何 使 用 微 秒 单位 ? 


答 : 有 些 时 候 ， 某 些 应 用 要 求 使 用 比 秒 更 小 的 时 间 单 位 来 表示 时 间 。 比 如 ， 在 一 段 测试 
程序 运行 的 程序 中 ， 可 能 要 使 用 到 微 秒 级 的 时 间 单 位 来 表示 时 间 。 如 果 需 要 微 秒 ， 只 需要 使 
用 函数 microtime(true) 即 可 。 

例如 : 


<?php 

$timestamp = microtime (true); 
echo $timestamp; 

D> 


返回 的 结果 为 “1315560215.7656”。 时 间 戳 精确 到 小 数 点 后 4 位 。 
疑问 2: 定义 日 期 和 时 间 时 出 现 警 告 怎么 办 ? 


答 : 在 运行 PHP 程序 时 ， 可 能 会 出 现 这 样 的 警告 : PHP Warning: date0: It is not safe to 
rely on the system's timezone settings 等 。 出 现 上 述 警 告 是 因为 PHP 所 取 的 时 间 是 格林 尼 治标 
准时 间 ， 所 以 与 用 户 当地 的 时 间 会 有 出 入 。 由 于 格林 尼 治 标准 时 间 与 北京 时 间 大 概 差 8 个 小 
时 左右 ， 所 以 会 弹出 警告 。 可 以 使 用 下 面 方法 中 的 任意 一 个 来 解决 。 

(1) 在 页 头 使 用 date_default_timezone_set0 设 置 默认 时 区 为 北京 时 间 ， 即 : 

<?php date_default_timezone_set ("PRC"); ?> 

如 本 章 例 10.2 所 示 。 


(2) 在 php.ini 中 设置 date.timezone 的 值 为 PRC， 设 置 语句 为 : date.timezone=PRC， 同 时 
取消 这 一 行 代码 的 注释 ， 即 去 掉 前 面 的 分 号 即 可 。 


第 11 章 
主流 的 编程 思想 
一 一 面向 对 象 
编程 


面向 对 象 编程 是 现在 编程 的 主流 ，PHP 也 不 例外 。 面 向 对 象 (object-oriented) 不 
同 于 面向 过 程 (process-oriented)， 它 用 类 、 对 象 、 关 系 、 属 性 等 提高 编程 的 效率 。 其 
主要 的 特性 是 可 封装 性 、 可 继承 性 和 多 态 性 。 本 章 主要 讲述 面向 对 象 的 相关 知识 。 
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11.1 类 和 对 象 的 介绍 


面向 对 象 编程 的 主要 好 处 就 是 把 编程 的 重心 从 处 理 过 程 转移 到 了 对 现实 世界 实体 的 表 
达 。 这 十 分 符合 人 们 的 普遍 思维 方法 。 

类 (Class) 和 对 象 (Object) 并 不 难 理解 。 试 想 一 下 ， 在 日 常生 活 中 ， 自 然 人 对 事物 的 认识 ， 
一 般 是 由 看 到 的 、 感 受到 的 实体 对 象 (日 常生 活 中 的 吃 、 穿 、 住 、 用 ) 归 纳 出 来 的 或 者 抽象 出 来 
的 类 特征 ， 这 就 是 人 们 认识 世界 的 过 程 。 

然而 ， 程 序 编写 者 需要 在 计算 机 的 世界 中 再 造 一 个 虚拟 的 “真实 世界 ”。 那 么 ， 在 这 
里 ， 程 序 员 就 要 像 “ 造 物 主 ” 一 样 思考 。 就 要 先 定 义 “ 类 ”， 然 后 再 由 “类 ”产生 一 个 个 
“实体 ”， 也 就 是 一 个 个 “对 象 ”。 

有 这 样 的 情况 : 过 年 的 时 候 ， 有 的 地 方 要 制作 “点 心 ”， 点 心 一 般 会 有 鱼 、 免 、 狗 等 生 
动 的 形状 。 而 这 些 不 同 的 形状 是 由 不 同 的 “模具 ”做 出 来 的 。 那 么 ， 在 这 里 ， 鱼 、 免 、 狗 的 
一 个 个 不 同 的 点 心 就 是 实体 ， 则 最 先 刻 好 的 “模具 ”就 是 类 。 要 明白 一 点 ， 这 个 “模具 ” 指 
的 是 被 刻 好 的 “形状 ”， 而 不 是 制作 “模具 ”的 材料 。 如 果 你 能 像 造物 主 一 样 用 意念 制作 出 
一 个 个 点 心 。 那 么 ， 你 的 意念 的 “形状 ”就 是 “模具 ”。 

OOP 是 面向 对 象 编程 (Object-oriented Programming) 的 英文 缩写 。 对 象 (Objecb 在 OOP 中 
是 由 属性 和 操作 组 成 的 。 属 性 (Attribute) 是 对 象 的 特性 或 是 与 对 象 关 联 的 变量 。 操 作 (Operation) 
是 对 象 中 的 方法 (Method) 或 函数 (Function)。 

由 于 OOP 中 最 为 重要 的 特性 之 一 就 是 可 封装 性 ， 所 以 对 Object 内 部 数据 的 访问 ， 只 能 通 
过 对 象 的 “操作 ”来 完成 ， 这 也 被 称 为 对 象 的 “接口 ”(Interfaces)。 

因为 类 是 对 象 的 模板 ， 所 以 类 描述 了 它 的 对 象 的 属性 和 方法 。 

面向 对 象 编程 具有 以 下 三 大 特点 。 

1) ”封装 性 

将 类 的 使 用 和 实现 分 开 管理 ， 只 保留 类 的 接口 ， 这 样 开发 人 员 就 不 用 去 知道 类 的 实现 过 
程 ， 只 需要 知道 如 何 使 用 类 即 可 ， 从 而 提高 了 开发 效率 。 

2) ”继承 性 

继承 是 面向 对 象 软件 技术 中 的 一 个 概念 。 如 果 一 个 类 A 继承 自 另 一 个 类 B， 就 把 这 个 A 
称 为 “B 的 子 类 ”， 而 把 B 称 为 “A 的 父 类 ”。 继 承 可 以 使 得 子 类 具有 父 类 的 各 种 属性 和 方 
法 ， 而 不 需要 再 次 编写 相同 的 代码 。 在 子 类 继承 父 类 的 同时 ， 可 以 重新 定义 某 些 属性 ， 并 重 
写 某 些 方法 ， 即 覆盖 父 类 的 原 有 属性 和 方法 ， 从 而 获得 与 父 类 不 同 的 功能 。 另 外 ， 还 可 以 为 
子 类 追加 新 的 属性 和 方法 。 继 承 可 以 实现 代码 的 可 重用 性 ， 简 化 了 对 象 和 类 的 创建 过 程 。 另 
外 ，PHP 支持 单 继 承 ， 也 就 是 一 个 子 类 只 能 有 一 个 父 类 。 

3) 多 态 性 

多 态 是 面向 对 象 程序 设计 的 重要 特征 之 一 ， 是 扩展 性 在 “继承 ”之 后 的 又 一 重大 表现 。 
同一 操作 作用 于 不 同类 的 实例 ， 将 产生 不 同 的 执行 结果 ， 即 不 同类 的 对 象 收 到 相同 的 消息 
时 ， 将 得 到 不 同 的 结果 。 





11.2 类 的 基本 操作 


类 是 面向 对 象 中 最 为 重要 的 概念 之 一 ， 是 面向 对 象 设计 中 最 基本 的 组 成 模块 。 类 可 以 简 
单 地 视 为 一 种 数据 结构 ， 在 类 中 的 数据 和 函数 称 为 类 的 成 员 。 


11.2.1 类 的 声明 
在 PHP 中， 声明 类 的 关键 字 是 class， 声 明 格式 如 下 : 


<?php 

权限 修饰 符 class 类 名 { 
类 的 内 容 ; 

} 


?> 

其 中 ， 权 限 修 饰 符 是 可 选项 ， 常 见 的 修饰 符 包括 public<、private 和 protected。 创 建 类 时 ， 
可 以 省 略 权 限 修饰 符 ， 此 时 默认 的 修饰 符 为 public。3 种 权限 修饰 符 的 区 别 如 下 。 

(1) 一 般 情况 下 ， 属 性 和 方法 默认 是 public 的 ， 这 意味 着 一 般 的 属性 和 方法 从 类 的 内 部 
和 外 部 都 可 以 访问 。 

(2) 用 关键 字 private 声明 的 属性 和 方法 ， 则 只 能 从 类 的 内 部 访问 。 也 就 是 说 ， 只 有 类 内 
部 的 方法 可 以 访问 用 此 关键 字 声 明 的 类 的 属性 和 方法 。 

(3) 用 关键 字 protected 声明 的 属性 和 方法 ， 也 是 只 能 从 类 的 内 部 访问 。 但 是 ， 通 过 “ 继 
承 ” 而 产生 的 “ 子 类 ”， 也 可 以 访问 这 些 属性 和 方法 。 

例如 ， 定 义 一 个 学 生 为 公共 类 ， 代 码 如 下 : 


public class Student { 


// 类 的 内 容 
了 


11.2.2 成 员 属性 


成 员 属 性 是 指 在 类 中 声明 的 变量 。 在 类 中 可 以 声明 多 个 变量 ， 所 以 对 象 中 可 以 存在 多 个 
成 员 属性 ， 每 个 变量 将 存储 不 同 的 对 象 属性 信息 。 

例如 以 下 定义 : 

public class Student { 

public $name; // 类 的 成 员 属 性 

上 

成 员 属性 必须 使 用 关键 词 进行 修饰 ， 常 见 的 关键 词 包括 public、protected、private 等 。 

如 果 没 有 特定 的 意义 ， 仍 然 需要 var 关键 词 修饰 。 另 外 ， 在 声明 成 员 属性 时 ， 可 以 不 进 
行 赋值 操作 。 


夷 蔚 小 妆 可 加 一 一 滋 记 前世 号 妆 咱 ”山上 | 小 加 
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11.2.3 成员 方法 


成 员 方法 是 指 在 类 中 声明 的 函数 。 由 于 在 类 中 可 以 声明 多 个 函数 ， 所 以 对 象 中 可 以 存在 
多 个 成 员 方 法 。 类 的 成 员 方法 可 以 通过 关键 字 进 行 修饰 ， 从 而 控制 成 员 方法 的 使 用 权限 。 

定义 成 员 方 法 的 例子 如 下 : 

class Student { 

public $name; // 类 的 成 员 属 性 

function GetIp(){ 

// 方 法 的 内 容 


. 
3 





11.2.4 类 的 实例 化 


面向 对 象 编程 的 思想 是 一 切 皆 为 对 象 。 类 是 对 一 个 事物 抽象 出 来 的 结果 ， 因 此 ， 类 是 抽 
象 的 。 对 象 是 某 类 事物 中 具体 的 那个 ， 因 此 ， 对 象 就 是 具体 的 。 例 如 ， 学 生 就 是 一 个 抽象 概 
念 ， 即 学 生 类 ， 但 是 姓名 叫 张 三 的 就 是 学 生 类 中 具体 的 一 个 学 生 ， 即 对 象 。 

类 和 对 象 可 以 描述 为 如 下 关系 。 类 用 来 描述 具有 相同 数据 结构 和 特征 的 “一 组 对 象 ”。 
“类 ”是 “对 象 ” 的 抽象 ， 而 “对 象 ”是 “类 ”的 具体 实例 ， 即 一 个 类 中 的 对 象 具 有 相同 的 
“型 ”， 但 其 中 每 个 对 象 却 具有 各 不 相同 的 “ 值 ”。 

3 类 是 具有 相同 或 相仿 结构 、 操 作 和 约束 规则 的 对 象 组 成 的 集合 ， 而 对 象 是 某 一 类 
起 的 具体 化 实例 ， 每 一 个 类 都 是 具有 某 些 共同 性 对 象 的 抽象 。 
类 的 实例 化 格式 如 下 : 
$ 变 量 名 = new 类 名 称 ( [参数 ] ) ; // 类 的 实例 化 
其 中 ，new 为 创建 对 象 的 关键 字 ，“$ 变 量 名 ”返回 对 象 的 名 称 ， 用 于 引用 类 中 的 方法 。 

参数 是 可 选 的 ， 如 果 存 在 参数 ， 则 用 于 指定 类 的 构造 方法 初始 化 对 象 使 用 的 值 。 如 果 没有 定 

义 构造 函数 参数 ，PHP 会 自动 创建 一 个 不 带 参 数 的 默认 构造 函数 。 


例如 下 面 的 例子 : 

class Student { 

public $name; // 类 的 成 员 属性 
function GetIp(){ 

// 方 法 的 内 容 ; 

. 

} 

$1ili = new Student(); // 类 的 实例 化 
$liufei = new Student(); // 类 的 实例 化 
$zhangming = new Student () 7 // 类 的 实例 化 
S$wangyi = new Student () 7 // 类 的 实例 化 


上 面 的 例子 实例 化 了 4 个 对 象 ， 并 且 这 4 个 对 象 之 间 没 有 任何 联系 ， 只 能 说 明 是 源 于 同 
一 个 类 。 可 见 ， 一 个 类 可 以 实例 化 多 个 对 象 ， 每 个 对 象 都 是 独立 存在 的 。 
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11.2.5 ”访问 类 中 的 成 员 属 性 和 方法 


通过 对 象 的 引用 ， 可 以 访问 类 中 的 成 员 属 性 和 方法 ， 这 需要 使 用 特殊 的 运算 符 “->”。 
具体 的 语法 格式 如 下 : 


$ 变 量 名 = new 类 名 称 () ; // 类 的 实例 化 

$ 变 量 名 -> 成 员 属性 = 值 ; / /为 成 员 属性 赋值 

$ 变 量 名 -> 成 员 属性 // 直 接 获取 成 员 的 属性 值 
$ 变 量 名 -> 成 员 方法 // 访 问 对 象 中 指定 的 方法 
另外 ， 程 序 员 还 可 以 使 用 以 下 特殊 的 访问 方法 。 

1. $this 


$this 存在 于 类 的 每 一 个 成 员 方 法 中 ， 它 是 一 个 特殊 的 对 象 引 用 方法 。 成 员 方 法 属于 哪个 
对 象 ，$this 引用 就 代表 哪个 对 象 ， 主 要 作用 是 专门 完成 对 象 内 部 成 员 之 间 的 访问 。 


2. 操作 符 “::” 
操作 符 “::” 可 以 在 没有 任何 声明 实例 的 情况 下 访问 类 中 的 成 员 。 使 用 的 语法 格式 如 下 : 
关键 字 : :变量 名 /常量 名 /方法 名 


其 中 关键 字 主要 包括 parent、self 和 类 名 3 种 。parent 关键 字 表 示 可 以 调用 父 类 中 的 成 员 
变量 、 常 量 和 成 员 方 法 。self 关键 字 表 示 可 以 调用 当前 类 中 的 常量 和 静态 成 员 。 类 名 关键 字 表 


示 可 以 调用 本 类 中 的 常量 、 变 量 和 方法 。 
下 面 通过 例子 介绍 类 的 声明 和 实例 生成 ， 其 中 将 描述 网 上 商城 的 客户 信息 ， 包 括 客户 姓 


名 和 订单 价格 。 
【 例 11.1】 类 的 声明 和 实例 生成 (示例 文件 chll\11.1.php)。 


<?php 

class guests{ 

private $name; 

private $orderprice; 

function setname ($name){ 
$this->name = $name; 

} 

function getname (){ 

return $this->name; 

function setorderprice ($orderprice){ 
$this->orderprice= $orderprice; 


本 


前 苏 淖 妆 避 副 一 一 漳 纪 前 世 吕 密 咱 ” 刷 上 1 洲 品 


function getorderprice (){ 

return S$this->orderprice; 

] 7 

$aa = new guests; 

$aa->setname (" 张 飞 "); 

$aa->setorderprice ("126 元 "); 

$bb =new guests; 

$bb->setname ("王蒙 "); 

$bb->setorderprice ("345 元 "); 

echo $aa->getname()."\t".$aa->getorderprice ()."<br/>"; 
echo $bb->getname()."\t".$bb->getorderprice (); 
> 


. 
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程序 运行 结果 如 图 11-1 所 示 。 PE 
【案例 剖析 】 Re 
GD) 用 class 关键 字 声明 了 一 个 类 ， 这 个 类 的 名 称 是 。 至 经 天 

guests。 在 大 括号 内 写 入 类 的 属性 和 方法 。 其 中 private 商 和 天 测 本 





$name、private $orderprice 为 类 guests 的 自 有 属性 ， 用 
private 关键 字 声明 。 也 就 是 说 ， 只 有 在 类 内 部 的 方法 可 
以 访问 它们 ， 类 外 部 是 不 能 访问 的 。 

(2) function setname($name)、function getname()、function setorderprice ($orderprice)、 
function getorderprice 0 是 类 的 方法 ， 它 们 可 以 对 private $name、private $orderprice 这 两 个 属性 
进行 操作 。Sthis 是 对 类 本 身 的 引用 。 用 “->” 连 接 类 属性 ， 格 式 如 $this->name 、S$this-> 
orderprice。 

(3) 用 new 关键 字 生成 一 个 对 象 ， 格 式 为 Saa = new Classname， 它 的 对 象 名 是 $aa。 当 程 
序 通过 new 生成 一 个 类 guests 的 实例 ， 也 就 是 对 象 $aa 的 时 候 ， 对 象 $aa 就 拥有 了 类 guests 的 
所 有 属性 和 方法 。 然 后 就 可 以 通过 “接口 ”， 也 就 是 这 个 对 象 的 方法 ( 即 类 的 方法 的 副本 ) 来 对 
对 象 的 属性 进行 操作 。 

(4) 通过 接口 setname($name) 给 实例 $aa 的 Sname 属性 赋值 为 “ 张 飞 ”， 通 过 
setorderprice ($orderprice) 给 实例 $aa 的 Sorderprice 属性 赋值 为 “126 元 ”。 同 样 道理 ， 通 过 接口 
操作 了 实例 $bb 的 属性 。 最 后 通过 接口 getname()、getorderprice0) 返 回 不 同 的 两 个 实例 的 $name 
属性 和 S$orderprice 属性 ， 并 且 打 印 出 结果 。 


11-1 类 的 声明 和 实例 生成 


11.3 ”构造 方法 和 析 构 方法 


构造 方法 存在 于 每 个 声明 的 类 中 ， 主 要 作用 是 执行 一 些 初始 化 任务 。 如 果 类 中 没有 直接 
声明 构造 方法 ， 那 么 类 会 默认 地 生成 一 个 没有 参数 且 内 存 为 空 的 构造 方法 。 

在 PHP 中 ， 声 明 构造 方法 的 方式 有 两 种 ， 在 PHP 5 版 本 之 前 ， 构 造 方法 的 名 称 必 须 与 类 
名 相同 ; 从 PHP 5 版 本 开始 ， 构 造 方 法 的 方法 名 称 必须 是 两 个 下 划 线 开头 的 ， 即 
_ xonstruct。 具 体 的 语法 格式 如 下 : 


function __construct([mixed args]){ 


// 方 法 的 内 容 

一 个 类 只 能 声明 一 个 构造 方法 。 构 造 方法 中 的 参数 是 可 选 的 ， 如 果 没有 传 入 参数 ， 那 么 
将 使 用 默认 参数 为 成 员 变 量 进行 初始 化 。 

在 例 11.1 中 ， 对 实例 $aa 的 Sname 属性 赋值 时 还 需要 通过 使 用 接口 setname($name) 进 行 操 
作 ， 如 $aa->setname(" 张 飞 ")。 如 果 想 在 生成 实例 Saa 的 同时 就 对 此 实例 的 属性 Sname 进行 赋 
值 ， 该 怎么 办 呢 ? 

这 时 就 需要 构造 方法 _construct0 了 。 这 个 函数 的 特性 是 ， 当 通过 关键 字 new 生成 实例 的 
时 候 ， 就 会 被 调用 执行 ， 它 的 用 途经 常 就 是 对 一 些 属 性 进行 初始 化 ， 也 就 是 对 一 些 属性 进行 
初始 化 的 赋值 。 


下 面 通过 例子 介绍 构造 方法 的 使 用 方法 和 技巧 。 
【 例 11.2】 使 用 构造 方法 (示例 文件 ch11\11.2.php)。 


<?php 

class guestsi 

private $name; 

private S$orderprice; 

function _ construct ($name, $orderprice){ 
$this->name = $name; 

$this->orderprice =$orderprice; 

证 

function getname(){ 

return $this->name; 

} 

function getorderprice (){ 

return $this->orderprice; 

} 

和 

$aa = new guests (" 王 蒙 ","135 元 "); 

$bb = new guests(" 刘 飞 ", "835 元 "); 

echo $aa->getname()."\t".$aa->getorderprice ()."<br/>"; 
echo $bb->getname()."\t".$bb->getorderprice (); 











2> 

程序 运行 结果 如 图 11-2 所 示 。 [Bou ry 

要 记 住 的 是 ， 构 造 方法 是 不 能 返回 (return) 值 的 。 文人 有” 多。 豆 看 V) 收 太夫 [A| 工具 TD 得 凤 H 

有 构造 方法 ， 就 有 它 的 反面 “ 析 构 方法 ” 天 二 二 天 
(destructor)。 它 是 在 对 象 被 销毁 的 时 候 被 调用 执行 的 。 但 


是 因为 PHP 在 每 个 请 求 的 最 终 都 会 把 所 有 资源 释放 ， 所 
以 析 构 方法 的 意义 是 有 限 的 。 有 具体 使 用 的 语法 格式 如 下 : 图 11-2 使 用 构造 方法 

function destruct(){ 

// 方 法 的 内 容 ， 通 常 是 完成 一 些 在 对 象 销毁 前 的 清理 任务 

PHP 具有 垃圾 回收 机 制 ， 可 以 自动 清除 不 再 使 用 的 对 象 ， 从 而 释放 更 多 内 存 。 析 构 方法 
是 在 垃圾 回收 程序 执行 前 被 调用 的 方法 ， 是 PHP 编程 中 的 可 选 内 容 。 

不 过 ， 析 构 方法 在 某 些 特定 行为 中 还 是 有 用 的 ， 比 如 在 对 象 被 销毁 时 清空 资源 或 者 记录 
日 志 信息 。 

以 下 两 种 情况 中 ， 析 构 方法 可 能 被 调用 执行 。 

(1) 代码 运行 时 ， 当 所 有 的 对 于 某 个 对 象 的 reference( 引 用 ) 被 毁 掉 的 情况 下 。 

(2) 当代 码 执行 到 最 终 ， 并 且 PHP 停止 请 求 的 时 候 。 








11.4 访 问 器 


另外 一 个 很 好 用 的 函数 是 访问 方法 (accessor)， 又 称 访问 器 。 由 于 OOP 思想 并 不 鼓励 直接 
从 类 的 外 部 访问 类 的 属性 ， 以 强调 封装 性 ， 所 以 可 以 使 用 _get 和 _ set 方法 来 达到 此 目的 。 
无 论 何 时 ， 类 属性 被 访问 和 操作 时 ， 访 问 方法 都 会 被 激发 。 通 过 使 用 它们 ， 可 以 避免 直接 对 
类 属性 的 访问 。 


CT 
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下 面 通过 例子 介绍 访问 器 的 使 用 方法 和 技巧 。 
【 例 11.3】 使 用 访问 器 (示例 文件 ch11\11.3.php)。 


<?php 

class guests{ 

public $property; 

function _ set($propName, $propValue){ 
$this->$propName = $propValue; 

} 

function _ get($propName){ 

return $this->$propName; 

， 

] 7 

$aa = new guests; 

$aa->name = " 刘 明 明 "; 

$aa->orderprice = "126 元 "; 

$bb = new guests; 

$bb->name = " 王 鹏 飞 "; 

$bb->counts="13 件 "; 

$bb-> orderprice= "365 元 "; 

echo $aa->name." 的 订单 总 价格 为 : " .$aa->orderprice."<br />"; 
echo $bb->name .” 的 商品 个 数 为 : " .$bb->counts 
."， 订 单 总 价格 为 : ".$bb->orderprice."<br />"; 
区 区 


程序 运行 结果 如 图 11-3 所 示 。 





@ 回合 epyecahe- P - | @ ocaihon 站 
文 作品。 移 各 日 查看 M)。 收 羡 六/A) 工具 和 0H) 

刘 明 明 的 订单 总 价格 为 : 126 元 

王 鹏 飞 的 商品 个 数 为 ,13 件 ， 订 单 总 价格 为 : 365 元 
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11-3 ”使 用 访问 器 
【案例 剖析 】 


(1) $aa 为 类 guests 的 实例 。 直 接 添加 属性 name 和 orderprice， 并 且 赋 值 。 例 如 $aa- 
>name = " 刘 小 帅 "和 $aa-> orderprice = "126 元 "， 此 时 ， 类 guests 中 的 _set 函数 被 调用 。$bb 实 
例 为 同样 的 过 程 。 另 外 ，S$bb 实例 添加 了 一 个 对 象 属性 counts。 

(2) echo 命令 中 使 用 到 的 对 象 属性 ， 如 $aa->name 等 ， 则 是 调用 了 类 guests 中 的 _get 函数 。 

(3) 此 例 中 ，_set 方 法 的 语法 格式 如 下 : 

function _set($propName, $propValue){ 


$this->$propName = $propValue; 
k 


_get 方法 的 语法 格式 如 下 : 


function _get($propName){ 
return S$this->$propName; 


其 中 ，$propName 为 “属性 名 ”，S$propValue 为 “属性 值 ”。 
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11.5 类 的 继承 


继承 (Inheritance) 是 OOP 中 最 为 重要 的 特性 与 概念 。 父 类 拥有 其 子 类 的 公共 属性 和 方法 。 
子 类 除了 拥有 父 类 具有 的 公共 属性 和 方法 外 ， 还 拥有 自己 独 有 的 属性 和 方法 。 

PHP 使 用 关键 字 extends 来 确认 子 类 和 父 类 ， 实 现 子 类 对 父 类 的 继承 。 

具体 的 语法 格式 如 下 : 


class 子 类 名 称 extends 父 类 名 称 { 

// 子 类 成 员 变 量 列表 

function 成 员 方法 (){ // 子 类 成 员 方法 
// 方 法 内 容 

} 

} 


下 面 通过 例子 介绍 类 的 继承 方法 。 
【 例 11.4】 使 用 类 继承 (示例 文件 ch11\11.4.php)。 


<?php 

class Vegetables{ 

var $tomato = "西红柿 "; // 定 义 变量 

var $cucumber = "黄瓜 "; 

}; 

class VegetablesType extends Vegetables{ // 类 之 间 继 承 
var $potato = "马铃薯 "; // 定 义 子 类 的 变量 
var $radish = " 茵 上 下"; 

}; 

$vegetables = new VegetablesType(); // 实 例 化 对 象 
echo "我 最 爱 的 蔬菜 : " .$vegetables->tomato.", ".$vegetables->cucumber 
.", ".$vegetables-> potato." , ".$vegetables-> radish; 

> 


程序 运行 结果 如 图 11-4 所 示 。 
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【案例 剖析 】 


从 结果 可 以 看 出 ， 本 例 创建 了 一 个 蔬菜 父 类 ， 子 类 通过 关键 字 extends 继承 了 蔬菜 父 类 中 
的 成 员 属 性 ， 最 后 对 子 类 进行 了 实例 化 操作 。 


11.6 ”抽象 类 和 接口 


抽象 类 和 接口 都 是 特殊 的 类 ， 因 为 它们 都 不 能 被 实例 化 。 下 面 主要 讲述 两 者 的 使 用 方法 
和 技巧 。 
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11.6.1 抽象 类 
抽象 类 只 能 作为 父 类 使 用 ， 因 为 抽象 类 不 能 被 实例 化 。 抽 象 类 使 用 关键 字 abstract 来 声 


明 ， 具 体 的 语法 格式 如 下 : 
abstract class 抽象 类 名 称 { 
// 抽 象 类 的 成 员 变 量 列表 
abstract function 成 员 方 法 1 (参数 ) ; // 抽 象 类 的 成 员 方法 
abstract function 成 员 方法 2 (参数 ) ; // 抽 象 类 的 成 员 方法 


抽象 类 与 普通 类 的 主要 区 别 在 于 ， 抽 象 类 的 方法 没有 方法 内 容 ， 而 且 至 少 包 含 一 个 抽象 
方法 。 另外， 抽象 方法 也 必须 使 用 关键 字 abstract 来 修饰 ， 而 且 抽象 方法 后 必须 有 分 号 。 
【 例 11.5】 使 用 抽象 类 (示例 文件 ch11\11.5.php)。 


<?php 

abstract class MyObject{ 

abstract function service($getName, $price, $num); 
class MyGoods extends MyObject{ 

function service($getName, $price, $num){ 

echo ' 您 购买 的 商品 是 ' .$getName .'， 该 商品 的 价格 是 : ' .$price.' 元 。'; 
echo ' 您 购买 的 数量 为 : ' .$num.， 件 。'; 

» 

. 

class MyComputer extends MyObject{ 

function service($getName, $price, $num){ 

echo ' 您 购买 的 商品 是 ' . $getName . '， 该 商品 的 价格 是 : ' . $price.' 元 。'; 
echo ' 您 购买 的 数量 为 :' .$num.， 件 。'; 

} 

J 

$book = new MyGoods (); 

$computer = new MYComputer () 7 

$book -> service(' 海 尔 洗 衣 机 ',3200,2); 

echo '<p>'; 

$computer -> service(' 创 维 电视 ',12000,10); 

> 


程序 运行 结果 如 图 11-5 所 示 。 





BO- he//ocalhost/code, P ~ © Bocalhest 
ME) 二 看 VM) 收藏 夫 (A)】 ”工具 (D) 者 助 (H) 


买 的 商品 是 海尔 洗衣 机 ， 该 商品 的 价格 是 : 3200 元 。 您 购买 的 数量 为 : 2 件 。 
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11-5 ”使 用 抽象 类 


11.6.2 ”接口 


继承 特性 简化 了 对 象 、 类 的 创建 ， 增 加 了 代码 的 可 重用 性 。 但 是 PHP 只 支持 单 继 承 ， 如 
果 想 实现 多 继承 ， 就 需要 使 用 接口 。PHP 可 以 实现 多 个 接口 。 

接口 类 通过 关键 字 interface 来 声明 。 接 口中 不 能 声明 变量 ， 只 能 使 用 关键 字 const 声明 为 
常量 的 成 员 属 性 。 接 口中 声明 的 方法 必须 是 抽象 方法 ， 并 且 接 口中 所 有 的 成 员 都 必须 是 public 
的 访问 权限 。 
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具体 的 使 用 语法 格式 如 下 : 六 
interface 接口 名 称 { // 使 用 interface 关键 字 声明 接口 
// 常 量 成 员 // 接 口中 的 成 员 只 能 是 常量 编 
// 抽 象 方法 // 成 员 方法 必须 是 抽象 方法 程 


与 继承 使 用 extends 关键 字 不 同 的 是 ， 实 现 接口 使 用 的 是 implements 关键 字 : 
class 实现 接口 的 类 implements 接口 名 称 


实现 接口 的 类 必须 实现 接口 中 声明 的 所 有 方法 ， 除 非 这 个 类 被 声明 为 抽象 类 。 
【 例 11.6】 类 之 间 的 继承 关系 及 接口 应 用 (示例 文件 chll\11.6.php)。 


<?php 
interface Intgoods{ 

// 这 两 个 方法 必须 在 子 类 中 继承 ， 修 饰 符 必须 为 public 
public function getName () 7 
public function getPrice(); 

} 

class goods implements Intgoods{ 
private $name = ' 海 尔 洗衣 机 '; 
private $price = '2888 元 '; 

// 具 体 实现 接口 声明 的 方法 
public function getName (){ 

return $this->name; 

} 
public function getPrice()1{ 

return $this->price; 


} 

// 这 里 还 可 以 有 自己 的 方法 

public function getOother(){ 
return ' 这 是 最 新 采购 的 商品 '; 





} 
E 
S$goods = new goods(); 
echo $goods->getName (); 
echo '<br/>'; 
echo $goo0ds->getPrice(); 
echo '<br/>'; 
echo $go0ds->getOother(); 
> 


程序 运行 结果 如 图 11-6 所 示 。 
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@ 国 -| 合 papyoceihoe- P - | 全 localhos 
文件 (月 ”六 筷 (E) ”可 看 (V)】 收藏 夫 (A) ”工具 (T) 帮助 (H) 


海尔 洗衣 机 








2888 元 
这 是 最 新 采购 的 商品 
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11-6 ”类 之 间 的 继承 关系 及 接口 应 用 





(1) 声明 接口 Intgoods， 然 后 定义 了 两 个 方法 getName0 和 getPrice0， 分 别 用 来 获取 商品 
的 名 称 和 价格 。 

(2) 实现 接口 的 类 goods 使 用 的 是 implements 关键 字 。 然 后 具体 实现 了 两 个 方法 
getName() 和 getPrice0。 另 外 ， 还 定义 了 自己 的 方法 getOther0。 

通过 上 面 的 学 习 ， 可 以 总 结 出 如 下 的 要 点 。 

(1) 在 PHP 中 ， 类 的 继承 只 能 是 单独 继承 ， 也 即 是 由 一 个 父 类 ( 基 类 ) 继 承 下 去 ， 而 且 可 以 
一 直 继 承 下 去 。PHP 不 支持 多 重 继承 ， 即 不 能 由 一 个 以 上 的 父 类 进行 继承 ， 也 即 是 类 C 不 能 
同时 继承 类 A 和 类 B。 

(2) 由 于 PHP 不 支持 多 重 继承 ， 为 了 对 特定 类 的 功能 的 拓展 ， 就 可 以 使 用 接口 (interface) 
来 实现 类 似 于 多 重 继承 的 好 处 。 接 口 用 interface 关键 字 来 声明 ， 并 且 单独 设立 接口 方法 。 

(3) 一 个 类 可 以 继承 于 一 个 父 类 ， 同 时 使 用 一 个 或 多 个 接口 。 类 还 可 以 直接 继承 于 某 个 特 
定 的 接口 。 

(4) 类 、 类 的 属性 和 方法 的 访问 ， 都 可 以 通过 放 在 属性 和 类 前 面 的 访问 修饰 符 进行 控制 。 
public 为 公共 属性 或 方法 ，private 为 私有 属性 或 方法 ，protected 为 受 保护 的 可 继承 属性 或 方法 。 

(5) 关键 字 final 放 在 特定 的 类 前 面 ， 表 示 此 类 不 能 再 被 继承 。final 放 在 某 个 类 方法 的 前 
面 ， 表 示 此 方法 不 能 在 继承 后 被 “ 覆 写 ”或 重新 定义 。 


NN 【案例 剖析 】 


11.7 ”面向 对 象 的 多 态 性 


多 态 性 是 指 同一 操作 作用 于 不 同类 的 实例 ， 将 产生 不 同 的 执行 结果 ， 即 不 同类 的 对 象 收 
到 相同 的 消息 时 ， 得 到 不 同 的 结果 。 在 PHP 中 ， 实 现 多 态 的 方法 有 两 种 ， 包 括 通过 继承 实现 
多 态 和 通过 接口 实现 多 态 。 


11.7.1 通过 继承 实现 多 态 


通过 继承 可 以 实现 多 态 的 效果 。 下 面 通过 例子 来 理解 实现 多 态 的 方法 。 
【 例 11.7】 通 过 继承 实现 多 态 (示例 文件 chll\11.7.php)。 


<?php 
abstract class Vegetables{ // 定 义 抽象 类 Vegetables 
abstract function go Vegetables(); // 定 义 抽象 方法 go_Vegetables 
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了 
class Vegetables_potato extends Vegetables{ // 马 铃 暮 类 继承 蔬菜 类 


public function go Vegetables(){ // 重 写 抽象 方法 
echo "我 们 开始 种 植 马铃薯 "”; // 输 出 信息 
: 


class Vegetables_radish extends Vegetables{ // 葛 卜 类 继承 蔬菜 类 


public function go Vegetables(){ // 重 写 抽象 方法 
echo "我 们 开始 种 植 萝卜” ; 
. 
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function change ($0bj){ // 自 定义 方法 根据 对 象 调用 不 同 的 方法 入 
if($ob]j instanceof Vegetables) 1{ 对 
S$ob]j->go_Vegetables () 7 条 
jelsei{ 程 
echo " 传 入 的 参数 不 是 一 个 对 象 "; // 输 出 信息 


} 


j 

echo "实例 化 Vegetables_potato: "; 

change (new Vegetables potato()); // 实 例 化 Vegetables_potato 
echo "<br>"; 

echo "实例 化 Vegetables_ radish: "; 

change (new Vegetables_radish()); // 实 例 化 Vegetables_radish 
> 


程序 运行 结果 如 图 11-7 所 示 。 








-|@ tp//iccalho.- P ~ ¢ | @ localhost 
文件 内。 自强 (E) 下 看 (V) 收藏 交 从 ) 工具 (帮助 (H) 


实例 化 Vegetables_potato: 我 们 开始 种 植 马 铃 昔 
实例 化 Vegetables_radish， 我 们 开始 种 植 梦 小 
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图 11-7 ”通过 继承 实现 多 态 





【案例 剖析 】 


从 结果 可 以 看 出 ， 本 例 创 建 了 一 个 抽象 类 Vegetables， 用 于 表示 各 种 蔬菜 的 种 植 方法 ， 然 
后 让 子 类 继承 这 个 Vegetables。 


11.7.2 ”通过 接口 实现 多 态 


下 面 通过 接口 的 方式 实现 与 例 11.7 一 样 的 效果 。 
【 例 11.8】 通 过 接口 实现 多 态 (示例 文件 chll\11.8.php)。 


<?php 
interface Vegetables{ // 定 义 接口 vegetables 
public function go Vegetables(); // 定 义 接口 方法 


} 

//Vegetables_potato 实现 Vegetables 接口 

class Vegetables_potato implements Vegetablesi{ 

public function go Vegetables(){ // 定 义 go_Vegetables 方法 
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echo "我 们 开始 种 植 马铃薯” ; // 输 出 信息 
} 

» 

//Vegetables_radish 实现 Vegetables 接口 

class Vegetables_radish implements Vegetablesi{ 





public function go Vegetables(){ // 定 义 go_Vegetables 方法 

echo "我 们 开始 种 植 萝 卜 ” ， // 输 出 信息 

+ 

} 

function change ($0obj){ // 自 定义 方法 根据 对 象 调用 不 同 的 方法 
if($obj instanceof Vegetables ){ 

$obj->go_Vegetables (); 

}elsei{ 

echo " 传 入 的 参数 不 是 一 个 对 象 "; // 输 出 信息 


. 


| 

echo "实例 化 Vegetables_potato: "; 

change (new Vegetables_potato()) 7 // 实 例 化 Vegetables_potato 
Scho <br> 

echo "实例 化 Vegetables_ radish: "7 

change (new Vegetables_ radish()); // 实 例 化 Vegetables_radish 
2 


程序 运行 结果 如 图 11-8 所 示 。 


加 -| 委 hepyecaho- PP- || 臣 Iocalhost 


文件 (站 锯 错 (E) 各 看 (V) 收藏 (A) 工具 (T) 帮 妈 (H) 


实例 化 Vegetables_potato: 我 们 开始 种 植 马铃薯 
实例 化 Vegetables_ radish: 我 们 开始 种 植 葛 卜 
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图 11-8 通过 接口 实现 多 态 





【案例 剖析 】 
从 结果 可 以 看 出 ， 本 例 创建 Vegetables 接口 ， 然 后 定义 一 个 空 方法 go_Vegetables()， 接 
着 定义 Vegetables_potato 和 Vegetables_radish 子 类 继承 Vegetables 接口 。 最 后 通过 instanceof 
关键 字 检 查 对 象 是 否 属于 Vegetables 接口 。 


11.8 PHP 7 的 新 变化 一 一 支持 匿名 类 


PHP 7 支持 通过 new class 来 实例 化 一 个 匿名 类 。 所 谓 匿名 类 ， 是 指 没 有 名 称 的 类 ， 只 能 
在 创建 时 用 new 语句 来 声明 它们 。 
下 面 通过 案例 来 对 比 PHP 7 和 PHP 7 之 前 版 本 的 区 别 和 联系 。 代 码 如 下 : 


<?php 
interface Logger { 

public function log(string $msg); 
: 


class Application { 
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private $logger; 


public function getLogger(): Logger { 
return $this->logger; 


} 


public function setLogger(Logger $logger) { 
$this->logger = $logger; 
} 
} 


$app = new Application; 
// 使 用 new class 创建 匿名 类 
$app->setLogger (new class implements Logger { 
public function logl(string $msg) { 
print ($msg); 
出 
]) 7 


$app->getLogger() ->1log ("北方 有 佳人 ， 绝 世 而 独立 。 一 顾 倾 人 城 ， 再 顾 倾 人 国 。 宁 不 知 倾城 与 
倾 国 ? 佳人 难 再 得 。") ; 


> 


程序 运行 结果 如 图 11-9 所 示 。 


-Bep//ocaho-. P - ©| Bcahoe 
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北方 有 佳人 ， 绝世 而 独立 .一 顾 倾 人 城 ， 再 兢 倾 人 国 。 
宁 不 知 倾城 与 合 困 ? 佳人 难 再 得 。 





图 11-9 使 用 匿名 类 


11.9 疑难 解 惑 


疑问 1: 如 何 理 解 “(a <b)?a:b” 的 含义 ? 

答 : 这 是 条 件 控 制 语句 ， 是 让 语句 的 单行 表示 法 。 它 的 语法 格式 如 下 : 

(条 件 判断 语句 ) ? 判断 为 true 的 行为 : 判断 为 false 的 行为 

站 语句 的 单行 表示 方式 的 好 处 是 ， 可 以 直接 对 条 件 判 断 的 结果 的 返回 值 进行 处 理 。 例 
如 ， 可 以 直接 把 返回 值 赋值 给 变量 : $variable = (a<b)? a : b， 如 果 a<b 的 结果 为 tue， 则 此 语 
句 返回 a， 并 且 直接 赋值 给 $variable， 如 果 a<b 的 结果 为 false， 则 此 语句 返回 b， 并 且 直 接 赋 
值 给 $variable。 

这 种 表示 方法 可 以 节约 代码 的 输入 量 ， 更 重要 的 是 可 以 提高 代码 执行 的 效率 。 由 于 PHP 
代码 执行 是 对 代码 由 上 至 下 的 一 个 过 程 ， 所 以 代码 的 行 数 越 少 ， 越 能 节约 代码 读 取 的 时 间 。 
像 这 样 在 一 行 语句 中 就 能 对 情况 做 出 判断 ， 并 且 对 代码 返回 值 进行 处 理 ， 无 疑 是 一 种 效率 相 
当 高 的 代码 组 织 方式 。 


前 苦 小 妆 可 加 一 一 滋 启 前 蔚 吕 涂 咱 ”山上 | 小 前 
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疑问 2: 如 何 区 分 抽象 类 和 类 的 不 同 之 处 ? 

答 : 抽象 类 是 类 的 一 种 ， 通 过 在 类 的 前 面 增加 关键 字 abstract 来 表示 。 抽 象 类 是 仅仅 用 来 
继承 的 类 。 通 过 abstract 关键 字 声明 ， 就 是 告诉 PHP， 这 个 类 不 再 用 于 生成 类 的 实例 ， 仅 仅 是 
用 来 被 其 子 类 继承 的 。 可 以 说 ， 抽 象 类 只 关注 于 类 的 继承 。 抽 象 方法 就 是 在 方法 前 面 添加 关 
键 字 abstract 声明 的 方法 。 抽 象 类 中 可 以 包含 抽象 方法 。 一 个 类 中 只 要 有 一 个 方法 通过 关键 字 
abstract 声明 为 抽象 方法 ， 则 整个 类 都 要 声明 为 抽象 类 。 然 而 ， 特 定 的 某 个 类 即便 不 含 抽象 方 
法 ， 也 可 以 通过 abstract 声明 为 抽象 类 。 
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第 12 章 
不 可 避免 的 问题 
背 误 处 理 和 
异常 处 理 





当 PHP 代码 运行 时 ， 会 发 生 各 种 错误 : 可 能 是 语法 错误 (通常 是 程序 员 造 成 的 
编码 错误 ); 可 能 是 缺少 功能 (由 于 浏览 器 差异 ); 可 能 是 由 于 来 自 服务 器 或 用 户 的 错 
误 输 出 而 导致 的 错误 ; 当然 ， 也 可 能 是 由 于 许多 其 他 不 可 预知 的 因素 导致 的 错误 。 
本 章 主要 讲述 错误 处 理 和 异常 处 理 。 
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12.1 常见 的 错误 和 异常 


错误 和 异常 是 编程 中 经 常 出 现 的 问题 。 下 面 主要 介绍 常见 的 错误 和 异常 。 
1. 拼写 错误 
拼写 代码 时 要 求 程序 员 非 常 仔细 ， 并 且 对 编写 完成 的 代码 还 需要 认真 地 去 检查 ， 否 则 会 
出 现 不 少 编写 上 的 错误 。 
另外 ，PHP 中 常量 和 变量 都 是 区 分 大 小 写 的， 例如 把 变量 名 abe 写成 ABC， 就 会 出 现 语 
NN 法 错误 。PHP 中 的 函数 名 、 方 法 名 、 类 名 不 区 分 大 小 写 ， 但 建议 使 用 与 定义 时 相同 的 名 字 。 
魔术 常量 不 区 分 大 小 写 ， 但 是 建议 全 部 大 写 ， 包 括 _LINE 、_FILE 、_DIR 、 
FUNCTION _ 、_CLASS_ 、 METHOD 、 NAMESPACE 。 知 道 了 这 些 规则 ， 程 序 
员 就 可 以 避免 大 小 写 的 错误 。 
另外 ， 编 写 代 码 有 时 需要 输入 中 文字 符 ， 编 程 人 员 容 易 在 输 完 中 文字 符 后 忘记 切换 输入 
法 ， 从 而 导致 输入 的 小 括号 、 分 号 或 者 引号 等 出 现 错误 ， 当 然 ， 这 种 错误 输入 在 大 多 数 编程 
软件 中 显示 的 颜色 会 跟 正确 的 输入 显示 的 颜色 不 一 样 ， 较 容易 发 现 ， 但 还 是 应 该 细心 谨慎 ， 
来 减少 错误 的 出 现 。 


2. 单 引 号 和 双 引 号 的 混乱 


单 引号 、 双 引号 在 PHP 中 没有 特殊 的 区 别 ， 都 可 以 用 来 创建 字符 串 。 但 是 必须 使 用 同一 
种 单 引号 或 双 引 号 来 定义 字符 串 ， 例 如 ，'Hello' 和 'Hello' 为 非法 的 字符 串 定义 。 单 引号 串 和 双 
引号 串 在 PHP 中 的 处 理 是 不 同 的 。 双 引号 串 中 的 内 容 可 以 被 解释 而 且 蔡 换 ， 而 单 引 号 串 中 的 
内 容 总 被 认为 是 普通 字符 。 

另外 ， 缺 少 单 引号 或 者 双 引号 也 是 经 常 出 现 的 问题 。 例 如 : 

echo "错误 处 理 的 方法 ; 


其 中 缺少 了 一 个 双 引 号 ， 运 行 时 会 提示 错误 。 
3. 括号 使 用 混乱 


首先 需要 说 明 的 是 ， 在 PHP 中 ， 括 号 包含 两 种 语义 ; 可 以 是 分 隔 符 ， 也 可 以 是 表达 式 。 
例如 : 

(1) 作为 分 隔 符 比较 常用 ， 比 如 (1+4)*4 等 于 20。 

(2) 在 (function0f7)0 中 ， 最 后 面 的 括号 表示 立即 执行 这 个 方法 。 

由 于 括号 的 使 用 层次 比较 多 ， 所 以 可 能 会 导致 括号 不 匹配 的 错误 。 

例如 : 


if((($a==$b)and ($b==$c))and ($c==$d) { // 此 处 缺少 一 个 括号 
echo "正确 的 括号 使 用 方法 ! " 
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4. 等 号 与 赋值 混淆 
等 号 与 赋值 符号 混淆 的 这 种 错误 一 般 较 常 出 现在 站 语句 中 ， 而 且 这 种 错误 在 PHP 中 不 会 
产生 错误 信息 ， 所 以 在 查找 错误 时 往往 不 容易 被 发 现 。 例 如 : 


if(s=1) 
echo ("没有 找到 相关 信息 ") ; 


上 述 代 码 在 逻辑 上 是 没有 问题 的 ， 它 的 运行 结果 是 将 1 赋值 给 了 s， 成 功 后 则 弹出 对 话 
框 ， 而 不 是 对 s 和 1 进行 比较 ， 这 不 符合 开发 者 的 本 意 。 正 确 写法 是 s==1， 而 不 是 s=1。 

5. 缺少 美元 符号 

在 PHP 中 ， 设 置 变量 时 需要 使 用 美元 符号 “$”， 如 果 不 添加 美元 符号 ， 就 会 引起 解析 
错误 


尊 育 球 和 过 月 育 消 划一 一 闹 瑟 否 浊 入 局 二 z 当 看 


例如 : 
for($s=1; $s<=10; s++) 1{ // 缺 少 一 个 变量 的 美元 符号 
2 (" 缺 少 美元 符号 ! ") ; 


需要 修改 s++ 为 gs+t+。 如 果 $s<=10 缺少 美元 符号 ， 则 会 进入 无 限 循环 状态 。 

6. 调用 不 存在 的 常量 和 变量 

如 果 调 用 没有 声明 的 常量 或 者 变量 ， 将 会 触发 NOTICE 错误 。 例 如 在 下 面 的 代码 中 ， 输 
出 时 错误 书写 了 变量 的 名 称 : 


<?php 
$abab = "错误 处 理 的 方法 " 
echo $abba; // 调 用 了 不 存在 的 变量 


?> 


如 果 运 行程 序 ， 会 提示 如 图 12-1 所 示 的 错误 。 
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12-1 调用 了 不 存在 的 变量 





7. 调用 不 存在 的 文件 

如 果 调 用 不 存在 的 文件 ， 程 序 将 会 停止 运行 。 例 如 : 

<?php 

include ("mybook.txt"); // 调 用 了 一 个 不 存在 的 文件 


i 


程序 运行 后 ， 将 会 弹出 如 图 12-2 所 示 的 错误 提示 信息 。 
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图 12-2 调用 了 不 存在 的 文件 





8. 环境 配置 的 错误 


如 果 环 境 配 置 不 当 ， 也 会 给 运行 带 来 错误 。 例 如 操作 系统 、PHP 配置 文件 和 PHP 的 版 本 
等 ， 这 些 如 果 配 置 不 正确 ， 将 会 提示 文件 无 法 打开 、 操 作 权限 不 具备 和 服务 器 无 法 连接 等 错 
误 信 息 。 

首先 ， 不 同 的 操作 系统 采用 不 同 的 路 径 格式 ， 这 些 都 会 导致 程序 运行 错误 。 此 外 ，PHP 
在 不 同 的 操作 系统 上 的 功能 也 会 有 差异 ， 数 据 库 的 运行 也 会 在 不 同 的 操作 系统 中 有 问题 出 现 
等 。 另 外 ，PHP 的 配置 也 很 重要 ， 由 于 各 个 计算 机 的 配置 方法 不 尽 相 同 ， 当 程序 的 运行 环境 
发 生变 化 时 ， 也 会 出 现 这 样 或 者 那样 的 问题 。 最 后 ， 是 PHP 的 版 本 问题 ，PHP 的 高 版 本 在 一 
定 程度 上 可 以 兼容 低 版 本 ， 但 是 针对 高 版 本 编写 的 程序 拿 到 低 版 本 中 运行 时 ， 会 出 现 意 想 不 
到 的 问题 ， 这 些 都 是 有 关 环 境 配置 的 不 同 而 引起 的 错误 。 


9. 数据 库 服务 器 连接 错误 


由 于 PHP 应 用 于 动态 网 站 的 开发 ， 所 以 经 常会 对 数据 库 进行 基本 的 操作 。 在 操作 数据 库 
之 前 ， 需 要 连接 数据 库 服 务 。 如 果 用 户 名 或 者 密码 设置 不 正确 ， 或 者 数据 库 不 存在 ， 或 者 数 
据 库 的 属性 不 允许 访问 等 ， 都 会 在 程序 运行 中 出 现 错误 。 

例如 以 下 的 代码 ， 在 连接 数据 库 的 过 程 中 ， 密 码 编写 是 错误 的 : 

<?php 


$conn = mysqli_connect ("localhost","root", "root"); // 连 接 MysQL 服务 器 
人 > 


程序 运行 后 ， 将 会 弹出 如 图 12-3 所 示 的 错误 提示 信息 。 
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12.2 错误 处 理 
常见 的 错误 处 理 方法 包括 使 用 错误 处 理 机 制 ， 使 用 DIE 语句 调试 、 自 定义 错误 和 错误 触 
发 器 等 。 下 面 讲述 如 何 处 理 程序 中 的 错误 。 
12.2.1 php.ini 中 的 错误 处 理 机 制 


在 前 面 的 例子 中 ， 错 误 提 示 会 显示 错误 的 信息 、 错 误 文件 的 行 号 信息 等 ， 这 是 PHP 最 基 
本 的 错误 报告 机 制 。 此 外 ，php.ini 文件 规定 了 错误 的 显示 方式 ， 包 括 配置 选项 的 名 称 、 默 认 
值 、 表 述 的 含义 等 。 常 见 的 错误 配置 选项 的 内 容 如 表 12-1 所 示 。 


表 12-1 常见 的 php-ini 文件 中 控制 错误 显示 的 配置 选项 含义 





半 瑟 否 浊 盘 习 半山 Z 小 中 





闹 谊 趟 各 普 肝 这 六 划 


名 称 默认 值 含义 

设置 错误 作为 PHP 的 一 部 分 输出 。 开 发 的 过 程 中 可 以 采用 默认 的 
设置 ， 但 是 为 了 安全 考虑 ， 在 生产 环境 中 还 是 设置 为 Off 比较 好 
这 个 设置 会 显示 所 有 的 出 错 信 息 。 这 种 设置 会 让 一 些 无 害 的 提示 也 
error_reporting Eall 会 显示 ， 所 以 可 以 设置 error_reporting 的 默认 值 : error_reporting = 
E ALL& ~E_NOTICE， 这 样 只 会 显示 错误 和 不 良 编码 

设置 记录 错误 日 志 的 文件 。 在 默认 情况 下 将 错误 发 送 到 Web 服务 


MAG8 


display_errors On 








Es J 器 日 志 ， 用 户 也 可 以 指定 写 入 的 文件 

html errors On 控制 是 否 在 错误 信息 中 采用 HTML 格式 

log errors Off 控制 是 否 应 该 将 错误 发 送 到 主机 服务 器 的 日 志文 件 
display startup_errors | Off 控制 是 否 显示 PHP 启动 时 的 错误 

track_errors Off 设置 是 否 保存 最 近 一 个 警告 或 错误 信息 


12.2.2 应 用 DIE 语句 进行 调试 


使 用 DIE 语句 进行 调试 的 优势 是 ， 不 仅 可 以 显示 错误 的 位 置 ， 还 可 以 输出 错误 信息 。 一 
旦 出 现 错误 ， 程序 将 会 终止 运行 ， 并 在 浏览 器 上 显示 出 错 之 前 的 信息 和 错误 信息 。 

前 面 曾经 讲述 过 ， 调 用 不 存在 的 文件 会 提示 错误 信息 ， 如 果 运 用 DIE 来 调试 ， 将 会 输出 
自 定义 的 错误 信息 。 

【 例 12.1】 应 用 DIE 语句 进行 调试 (示例 文件 ch12\12.1.php)。 

<?php 

if(!file exists ("wenjian.txt")){ 

qdie ("文件 不 存在 ") ; 

}elsef{ 

$file = fopen ("wenjian.txt","r"); 

和 


> 
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程序 运行 后 ， 结 果 如 图 12.4 所 示 。 - 二 
与 基本 的 错误 报告 机 制 相 比 ， 使 用 DIE 语句 调试 显得 ”人 @ Ee 5- See 
更 有 效 ， 这 是 由 于 它 采用 了 一 个 简单 的 错误 处 理 机 制 ， 在 。 Xf 
错误 之 后 终止 了 脚本 。 和 


12.2.3” 自 定义 错误 和 错误 触发 器 12-4 ”应 用 DIE 语句 进行 调试 


简单 地 终止 脚本 并 不 总 是 恰当 的 方式 。 下 面 讲 述 如 何 自 定义 错误 和 错误 触发 器 。 创 建 一 
个 自 定义 的 错误 处 理 器 非常 简单 ， 用 户 可 以 创建 一 个 专用 函数 ， 然 后 在 PHP 程序 发 生 错误 时 
调用 该 函数 。 

自 定义 的 错误 函数 的 语法 格式 如 下 : 

error_function(error_level,error message,error file,error line,error context) 

该 函数 必须 至 少 包 含 level 和 message 参数 ， 另 外 3 个 参数 fle、line 和 context 是 可 选 
的 。 各 个 参数 的 具体 含义 如 表 12-2 所 示 。 


表 12-2 各 个 参数 的 含义 


























参 数 含义 
error level 必需 参数 。 为 用 户 定义 的 错误 规定 错误 报告 级 别 。 必 须 是 一 个 值 
errOr message 必需 参数 。 为 用 户 定义 的 错误 规定 错误 消息 
error file 可 选 参数 。 规 定 错误 在 其 中 发 生 的 文件 名 
error line 可 选 参数 。 规 定 错误 发 生 的 行 号 


error_context 可 选 参数 。 规 定 一 个 数组 ， 包 含 了 当 错 误 发 生 时 在 使 用 的 每 个 变量 以 及 它们 的 值 


参数 error level 为 定义 错误 规定 的 报告 级 别 ， 这 些 错误 报告 级 别 是 错误 处 理 程序 将 要 处 
理 的 错误 的 类 型 。 具 体 的 级 别 值 和 含义 如 表 12-3 所 示 。 


表 12-3 错误 的 级 别 值 和 含义 

















数 值 常 量 含义 
2 E WARNING 非 致命 的 mm-time 错误 。 不 暂停 脚本 执行 
Run-time 通知 。 脚 本 发 现 可 能 有 错误 发 生 ， 但 也 可 能 在 脚本 正常 
呈 E_NOTICE ae 
到 运行 时 发 生 
致命 的 用 户 生 成 的 错误 。 类 似 于 程序 员 用 PHP 函数 trigger_error0 
256 E_USER ERROR 
设置 的 E ERROR 
非 致 命 的 用 户 生成 的 警告 。 这 类 似 于 程序 员 使 用 PHP 函数 
512 E USER WARNING 
一 trigger_error0 设 置 的 E WARNING 
用 户 生 成 的 通知 。 这 类 似 于 程序 员 使 用 PHP 函数 trigger_error0 设 
1024 E USER NOTICE 
置 的 E NOTICE 
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第 昌 
续 表 六 
常 量 入 灾 | 
可 捕获 的 致命 错误 。 类 似 E_ERROR， 但 可 被 用 户 定义 的 处 理 程序 络 
4096 E RECOVERABLE ERROR 1 
捕获 六 
E ALL 所 有 错误 和 警告 交 
下 面 通过 例子 来 讲解 如 何 自 定义 错误 和 错误 触发 器 。 下 
首先 创建 一 个 处 理 错误 的 函数 ; 央 
function customError($errno, S$errstr) 
! 
echo "<b> 错 误 : </b> [$errno] S$errstr<br />"; 和 
echo "终止 程序 "; 民 
die(); 常 
} 处 
理 


上 述 代码 是 一 个 简单 的 错误 处 理 函数 。 当 它 被 触发 时 ， 它 会 取得 错误 级 别 和 错误 消息 。 
然后 它 会 输出 错误 级 别 和 消息 ， 并 终止 程序 。 

创建 了 一 个 错误 处 理 函数 后 ， 下 面 需要 确定 在 何 时 触发 该 函数 。 在 PHP 中 ， 使 用 set_error_ 
handler0 函 数 来 设置 用 户 自 定义 的 错误 处 理 函 数 。 该 函数 用 于 创建 运行 期 间 的 用 户 自己 的 错误 处 
理 方法 。 该 函数 会 返回 旧 的 错误 处 理 程序 ， 若 失败 ， 则 返回 null。 具 体 的 语法 格式 如 下 : 

set_error handler (error_ function, error types) 

其 中 ，error function 为 必需 参数 ， 规 定 发 生 错 误 时 运行 的 函数 ; error types 是 可 选 参 
数 ， 如 果 不 选择 此 参数 ， 则 表示 默认 值 为 “E_ALL”。 

在 本 例 中 ， 针 对 所 有 错误 来 使 用 自 定 义 错误 处 理 程序 ， 具 体 代码 如 下 : 


set_error_ handler ("customError"); 


下 面 通过 尝试 输出 不 存在 的 变量 ， 来 测试 这 个 错误 处 理 程序 。 
【 例 12.2】 自 定义 错误 (示例 文件 ch12\12.2.php)。 


<?php 
/7 定义 错误 函数 
function customError($errno, $errstr){ 
echo "<b> 错 误 :</b> [$errno] S$errstr"; 


} 
// 设 置 错误 函数 的 处 理 
set_error handler ("customError"); 
// 触 发 自 定义 错误 函数 
echo($test); 
?> 


AG 





- 0O x 


程序 运行 后 ， 结 果 如 图 12-5 所 示 。 © [@-|B hp/hocalho.. D+ © | @ localhost 
在 脚本 中 用 户 输入 数据 的 位 置 设置 当 用 户 的 输入 xs ss® sav aasm IRD mm 
无 效 时 触发 错误 的 很 有 用 的 。 在 PHP 中 ， 这 个 任务 由 错误 : [8] Undefined variable: test 
trigger_error() 来 完成 。trigger_error() 函 数 创建 用 户 定义 A 
的 错误 消息 。 
trigger_error() 用 于 在 用 户 指 定 的 条 件 下 触发 一 
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错误 消息 。 它 与 内 建 的 错误 处 理 器 一 同 使 用 ， 也 可 以 与 由 set_error handler0 函 数 创建 的 用 户 自 
定义 函数 一 起 使 用 。 如 果 指 定 了 一 个 不 合法 的 错误 类 型 ， 该 函数 返回 false， 否 则 返回 true。 
trigger_error0 函 数 的 具体 语法 格式 如 下 : 


trigger_error (error message, error types) 


其 中 error_message 为 必需 参数 ， 规 定 错 误 消息 ， 长 度 限制 为 1024 个 字符 ; error types 为 
可 选 参数 ， 规 定 错误 消息 的 错误 类 型 ， 可 能 的 值 为 E USER ERROR、E USER WARNING 
或 者 E_USER_NOTICE 。 

【 例 12.3】 使 用 trigger_error0 函 数 (示例 文件 ch12\12.3.php)。 


<?php 
$test = 5» 
if ($test > 4){ 
trigger error("Value must be 4 or below"); 





?> 


程序 运行 后 ， 结 果 如 图 12-6 所 示 。 由 于 test 数值 为 5， 发 生 了 E_USER_WARNING 错误 。 
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图 12-6 使 用 trigger_error() 函 数 


下 面 通过 示例 来 讲述 trigger_error0 函 数 和 自 定义 函数 一 起 使 用 的 处 理 方法 。 
【 例 12.4】 使 用 自 定义 函数 和 trigger_error0 函 数 (示例 文件 ch12\12.4.php)。 
<?php 
// 定 义 错误 函数 
function customError($errno, $errstr){ 
echo "<b> 错 误 :</b> [$errno] $errstr"; 


} 
// 设 置 错误 函数 的 处 理 
set _error handler ("customError", E_ USER WARNING); 
// trigger_error 函数 
$test = 57 
if ($test>4){ 
trigger error ("Value must be 4 or below", E USER WARNING); 
} 


全 


程序 运行 后 ， 结 果 如 图 12-7 所 示 。 
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错误 : [512] Value must be 4 or below 
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12-7 ”使 用 自 定义 函数 和 trigger_error() 函 数 


12.2.4 ”错误 记录 


在 默认 情况 下 ， 根 据 在 php.ini 中 的 error_log 配置 ， PHP 向 服务 器 的 错误 记录 系统 或 文件 发 
送 错误 记录 。 通 过 使 用 error_ log0 函 数 ， 用 户 可 以 向 指定 的 文件 或 远程 目的 地 发 送 错 误 记 录 。 

通过 电子 邮件 向 用 户 自己 发 送 错误 消息 ， 是 一 种 获得 指定 错误 的 通知 的 好 办 法 。 下 面 通 
过 示例 来 讲解 。 

【 例 12.5】 通 过 E-mail 发 送 错误 信息 (示例 文件 ch12\12.5.php)。 

<?php 

// 定 义 错误 函数 

function customError ($errno, $errstr){ 

echo "<b> 错 误 :</b> [$errno] $errstr <br/>"; 

echo "错误 记录 已 经 发 送 完毕 "; 

error_log ("错误 : [$errno] $errstr",l1l, "someone@example.com", 

"From: webmastere@example.com"); 


月 皮球 扣 过 月 育 消 导 一 一 靖 瑟 否 地 陋习 习 jz 兴国 





时 

// 设 置 错误 函数 的 处 理 

set_error handler ("customError", E_USER WARNING); 
//trigger_error 函数 

Stest = 57 

if (Stest > 4){ 

trigger_error("Value must be 4 or below", E_USER WARNING); 
i 











2> 
程序 运行 后 ， 结 果 如 图 12-8 所 示 。 在 指定 的 someone@example.com 邮箱 中 将 收 到 错误 信息 。 
所 口 x 
人 [@-|@ repywecehe- P - © | @ vcathost x 
文件 (有 ] ”编辑 (E) ”下 看 (V) 收藏 夫 (A) 工具 (T) 者 助 (H) 
错误 : [512] Value must be 4 or below | 
错误 记录 已 经 发 送 完 毕 
v 
碑 125% ~ 
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12.3 PHP 7 新 变化 一 一 改变 了 错误 的 报告 方式 


PHP 7 改变 了 大 多 数 错误 的 报告 方式 。 不 同 于 PHP 5 的 传统 错误 报告 机 制 ， 现 在 大 多 数 
错误 被 作为 Error 异常 抛 出 。 

这 种 Error 异常 可 以 像 普 通 异 常 一 样 被 try / catch 块 所 捕获 。 如 果 没 有 匹配 的 try / catch 
块 ， 则 调用 异常 处 理 函 数 (set_exception_handler0) 进 行 处 理 。 如 果 尚 未 注册 异常 处 理 函 数 ， 则 
按照 传统 方式 处 理 : 被 报告 为 一 个 致命 错误 (Fatal Error)。 

Error 类 并 不 是 从 Exception 类 扩展 出 来 的 ， 所 以 用 catch (Exception $e) { … } 这 样 的 代 
码 是 捕获 不 到 Error 的 。 用 户 可 以 用 catch (Error $e) { … } 这 样 的 代码 ， 或 者 通过 注册 异常 处 
理 函 数 ( set_exception handler()) 来 捕获 Error。 

【 例 12.6】 用 try/catch 块 捕获 异常 (示例 文件 ch12\12.6.php)。 

<?php 

class Mathtions // 定 义 一 个 类 Mathtions 


protected $n = 10; // 定 义 变量 


// 求 余数 运算 ， 除 数 为 0， 抛 出 异常 
public function dotion(): string 


{ 





try { 
$value = $this->n % 07 
return $value; 
} catch (DivisionByZeroError S$e) { 
return $e->getMessage(); 
} 全 口 x 
} @ Btttpilocalhowcode P ~ | @ localhost 
} | ztP_ 时 下 WwW， 工具 TT) 碍 MtH) 


Modulo by zero 
$aa = new Mathtions(); 
print ($aa->dotion()); | 
> 


矶 125% = 





程序 运行 后 ， 结 果 如 图 12-9 所 示 。 129, 程 凡 二季 续 要 


12.4 异常 处 理 


异常 (Exception) 用 于 在 指定 的 错误 发 生 时 改变 脚本 的 正常 执行 流程 。 PHP 提供 了 一 种 新 
的 面向 对 象 的 异常 处 理 方法 。 下 面 主要 讲述 异常 处 理 的 方法 和 技巧 。 


12.4.1 异常 的 基本 处 理 方法 


异常 处 理 用 于 在 指定 的 错误 (异常 ) 情 况 发 生 时 改变 脚本 的 正常 执行 流程 。 当 异常 被 触发 
时 ， 通 常会 发 生 以 下 动作 。 

















Dd 
Gi 


(1) 当前 代码 状态 被 保存 。 

(2) 代码 执行 被 切换 到 预定 义 的 异常 处 理 器 函数 。 

(3) 根据 情况 ， 处 理 器 也 许 会 从 保存 的 代码 状态 重新 开始 执行 代码 ， 终 止 脚 本 执行 ， 或 
从 代码 中 另外 的 位 置 继 续 执行 脚本 。 

当 异 常 被 抛 出 时 ， 其 后 的 代码 不 会 继续 执行 ，PHP 会 尝试 查找 匹配 的 catch 代码 块 。 如 果 
异常 没有 被 捕获 ， 而 且 又 没有 使 用 set_exception handler0 做 相应 处 理 的 话 ， 那 么 将 发 生 一 个 
严重 的 错误 ， 并 且 输 出 Uncaught Exception( 未 捕获 异常 ) 的 错误 消息 。 

下 面 的 示例 中 抛 出 一 个 异常 ， 同 时 不 去 捕获 它 。 

【 例 12.7】 没 有 捕获 异常 (示例 文件 ch12\12.7.php)。 

<?php 

// 创 建 带 有 异常 的 函数 

function checkNum($number){ 


if($number>1){ 
throw new Exception("Value must be 1 or below"); 


有 


return true; 


人 
// 抛 出 异常 


checkNum(2); 
SC 
程序 运行 后 ， 结 果 如 图 12-10 所 示 。 由 于 没有 捕获 异常 ， 出 现 了 错误 提示 消息 。 


- OO x 





:Brulecahoweod P - OB echos 
HN) oF) ESV) ta TAM HI 








;> - 一 一 一 一 
imo [Function [Location 
0. 0004 351912| (ain C) [3.7. php:0 
2| 0 0004 351912]eheckNum( ) -12 7. php:10 
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图 12-10 没有 捕获 异常 
如 果 想 避免 出 现 上 面 例子 出 现 的 错误 ， 需 要 创建 适当 的 代码 来 处 理 异常 。 处 理 异 常 的 程 


序 应 当 包 括 下 列 代码 块 。 

(1) try 代码 块 。 使 用 异常 的 函数 应 该 位 于 try 代码 块 内 。 如 果 没 有 触发 异常 ， 则 代码 将 照 
常 继续 执 行 。 但 是 如 果 异 常 被 触发 ， 会 抛 出 一 个 异常 。 

(2) throw 代码 块 。 这 里 规定 如 何 触发 异常 。 每 一 个 throw 必须 对 应 至 少 一 个 catch。 

(3) catch 代码 块 。catch 代码 块 会 捕获 异常 ， 并 创建 一 个 包含 异常 信息 的 对 象 。 

【 例 12.8】 捕 获 异 常 (示例 文件 ch12\12.8.php)。 


<?php 
// 创 建 可 抛 出 一 个 异常 的 函数 


function checkNum($number){ 
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if($number>1){ 
throw new Exception ("数值 必 须 小 于 或 等 于 1"); 
} 


return true; 


} 

// 在 try 代码 块 中 触发 异常 

try{ 

checkNum (2); 

// 如 果 没 有 异常 ， 则 会 显示 以 下 信息 
echo ' 没 有 任何 异常 ' ; 





3 

// 捕 获 异 常 

catch (Exception $e){ 

echo ' 异 常 信息 : ' .$e->getMessage(); 


} 


?> 


程序 运行 后 ， 结 果 如 图 12-11 所 示 。 由 于 抛 出 异常 后 捕获 了 异常 ， 所 以 出 现 了 提示 消息 。 





- oO x 
@ 加 -| 总 Mp/focalho.. PD- Clocalhost 


文件 (站 沪 久 () 吾 看 V) 收 羔 关 (A) 工具 (帮助 (H) 
异常 信息 : 数值 必须 小 于 或 等 于 1 





R125% ~ ] 





图 12-11 捕获 了 异常 
【案例 剖析 】 


(1) 创建 checkNum0 函 数 ， 它 检测 数字 是 否 大 于 1。 如 果 是 ， 则 抛 出 一 个 异常 。 
(2) 在 try 代码 块 中 调用 checkNum0 函 数 。 

(3) checkNum0) 函 数 中 的 异常 被 抛 出 。 

(4) catch 代码 块 接收 到 该 异常 ， 并 创建 一 个 包含 异常 信息 的 对 象 ($e)。 

(5) 通过 从 这 个 exception 对 象 调用 $e->getMessage0， 输 出 来 自 该 异常 的 错误 消息 。 


12.4.2 自 定义 的 异常 处 理 器 


创建 自 定义 的 异常 处 理 程序 非常 简单 ， 只 需要 创建 一 个 专门 的 类 ， 当 PHP 程序 中 发 生 异 
常 时 ， 调 用 该 类 的 函数 即 可 。 当 然 ， 该 类 必须 是 exception 类 的 一 个 扩展 。 

这 个 自 定义 的 exception 类 继承 了 PHP 的 exception 类 的 所 有 属性 ， 然 后 用 户 可 向 其 添加 
自 定义 的 函数 。 

下 面 通过 例子 来 讲解 如 何 创建 自 定义 的 异常 处 理 器 。 

【 例 12.9】 创 建 自 定义 的 异常 处 理 器 (示例 文件 ch12\12.9.php)。 


<?php 
class customException extends Exception{ 


public function errorMessage(){ 
// 错 误 消息 


$errorMsg = ' 异 常 发 生 的 行 : '.$this->getLine().' in '.$this->getFile() 
-1: <b>'.$this->getMessage() .'</b> 不 是 一 个 有 效 的 邮箱 地 址 ' ; 


合 18 


return $errorMsg; 


$email = "someone@example.321lcom"; 
try 


{ 

// 检 查 是 否 符合 条 件 

if(filter var($email, FILTER VALIDATE EMAIL) === FALSE) { 
// 如 果 邮 件 地 址 无 效 ， 则 抛 出 异常 

throw new customException ($email); 

3 

} catch (customException $e){ 

// 显 示 自 定义 的 消息 

echo $e->errorMessage(); 





?> 
程序 运行 后 ， 结 果 如 图 12-12 所 示 。 
= 牢 
@ 七 .htpylocalhostjcr "CG 登 localhos 
文 #(P。 交锋 日 。 亚 理 (/) 收藏 关 (A) 工具 (T) 天 二 (H) 
异常 发 生 的 行 19 in C:\wamp\www\code\ch12\12. 9. php: 
someonefexample. 321com 不 是 一 个 有 效 的 邮箱 地 址 
成 125% ~ 
图 12-12 自 定义 异常 处 理 器 
【案例 剖析 】 


(1) customException0 类 是 作为 旧 的 exception 类 的 一 个 扩展 来 创建 的 ， 这 样 它 就 继承 了 


旧 类 的 所 有 属性 和 方法 。 
(2) 创建 errorMessage0 函 数 ， 如 果 E-mail 地 址 不 合法 ， 则 该 函数 返回 一 条 错误 消息 。 


(3) 把 Semail 变量 设置 为 不 合法 的 E-mail 地 址 字符 串 。 
(4) 执行 ty 代码 块 ， 由 于 E-mail 地 址 不 合法 ， 因 此 抛 出 一 个 异常 。 
(5) eatch 代码 块 捕获 异常 ， 并 显示 错误 消息 。 


12.4.3 ”处 理 多 个 异常 


在 上 面 的 案例 中 ， 只 是 检查 了 邮箱 地 址 是 否 有 效 。 如 果 用 户 想 检查 邮箱 是 否 为 雅虎 邮 
箱 ， 或 想 检 查 邮 箱 是 否 有 效 等 ， 这 就 出 现 了 多 个 可 能 发 生 异常 的 情况 。 用 户 可 以 使 用 多 个 
让 ...else 代码 块 ， 或 一 个 switch 代码 块 ， 或 者 嵌 套 多 个 异常 。 这 些 异常 能 够 使 用 不 同 的 
exception 类 ， 并 返回 不 同 的 错误 消息 。 


【 例 12.10】 处 理 多 个 异常 (示例 文件 ch12\12.10.php)。 


<?php 
class customException extends Exception{ 
public function errorMessage(){ 


// 定 义 错误 信息 


$errorMsg = ' 错 误 消 息 的 行 : ' . Sthis->getLine() .' in '.$this->getFile() 


.1: <b>'.$this->getMessage() .'</b> 不 是 一 个 有 效 的 邮箱 地 址 ' ; 
return S$errorMsg; 
时 
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$email = "someone@yahoo.com"; 


try{ 
// 检 查 是 否 符合 条 件 


if(filter var($email, FILTER VALIDATE EMAIL) === FALSE) 


{ 
// 如 果 邮 箱 地 址 无 效 ， 则 抛 出 异常 


throw new customException ($email); 


+. 

// 检 查 邮箱 是 否 是 雅虎 邮箱 

if(strpos ($email, "yahoo") !== FALSE){ 
throw new Exception("$email 是 一 个 雅虎 邮箱 ") ; 
3 

} catch (customException $e) { 

echo $e->errorMessage(); 

} catch (Exception $e) { 

echo $e->getMessage(); 


} 














?> 

程序 运行 后 ， 结 果 如 图 12-13 所 示 。 上 述 代码 测试 了 - oa x 
两 种 条 件 ， 如 果 任 何 条 件 都 不 成 立 ， 则 抛 出 一 个 异常 。 QO. El ee 

[案例 剖析 】 someoneQyahoo. com 是 一 个 雅虎 好 箱 | 


用 125% ~ 





(1) customException0 类 是 作为 旧 的 exception 类 的 一 

个 扩展 来 创建 的 ， 这 样 它 就 继承 了 旧 类 的 所 有 属性 和 方法 。 人 
(2) 创建 errorMessage0 函 数 。 如 果 E-mail 地 址 不 合法 ， 则 该 函数 返回 一 个 错误 消息 。 
(3) 执行 try 代码 块 ， 在 第 一 个 条 件 下 ， 不 会 抛 出 异常 。 
(4) 由 于 E-mail 含有 字符 串 “yahoo”， 第 二 个 条 件 会 触发 异常 。 
(5) catch 代码 块 会 捕获 异常 ， 并 显示 恰当 的 错误 消息 。 


12.4.4 设置 顶层 异常 处 理 器 


所 有 未 捕获 的 异常 都 可 以 通过 顶层 异常 处 理 器 来 处 理 。 顶 层 异 常 处 理 器 可 以 使 用 set_ 
exception_handler() 函 数 来 实现 。 

set_exception_handler() 函 数 设 置 用 户 自 定义 的 异常 处 理 函 数 。 该 函数 用 于 创建 运行 时 期 间 
的 用 户 自己 的 异常 处 理 方法 。 该 函数 会 返回 旧 的 异常 处 理 程序 ， 若 失败 ， 则 返回 null。 具 体 
的 语法 格式 如 下 : 


set_ exception handler (exception function) 


其 中 exception_function 参数 为 必需 的 参数 ， 规 定 未 捕获 的 异常 发 生 时 调用 的 函数 ， 该 函 
数 必须 在 调用 set_exception_handler() 函 数 之 前 定义 。 这 个 异常 处 理 函数 需要 一 个 参数 ， 即 抛 
出 的 exception 对 象 。 

【 例 12.11】 使 用 顶层 异常 处 理 器 (示例 文件 ch12\12.11.php)。 

<?php 


function myException ($exception){ 
echo "<b> 异 常 是 :</b> " ， $exception->getMessage(); 








set_exception handler('myException'); Te 
throw new Exception(' 正 在 处 理 未 被 捕获 的 异常 ') ; <€ [OE Ty Te 
2 E.R 








异常 是 : 正在 处 理 未 被 捕获 的 异常 





程序 运行 后 ， 结 果 如 图 12-14 所 示 。 上 述 代码 不 存在 
catch 代码 块 ， 而 是 触发 项 层 的 异常 处 理 程序 。 用 户 应 该 
使 用 此 函数 来 捕获 所 有 未 被 捕获 的 异常 图 12-14 使 用 项 层 异常 处 理 器 


12.5 ”案例 实战 一 一 处 理 异常 或 错误 


错误 处 理 也 叫 异 常 处 理 。 通 过 使 用 try...throw...catch 结构 和 一 个 内 置 函 数 Exception() 来 
“ 抛 出 ”和 “处 理 ” 错 误 或 异常 。 

下 面 通 过 打开 文件 的 例子 来 介绍 异常 的 处 理 方法 和 技巧 。 

【 例 12.12】 处 理 异常 或 错误 (示例 文件 ch12\12.12.php)。 


<?php 

$DOCUMENT_ROOT = $_SERVER['DOCUMENT ROOT']; 
@$fp = fopen("$DOCUMENT _ ROOT/book.txt",'rb'); 
try{ 

(Fp)t 

throw new Exception(" 文 件 路 径 有 误 或 找 不 到 文件 。") ; 
}catch (Exception $exception){ 

echo $exception->getMessage(); 

echo "在 文件 ". $exception->getFile() 

." 的 " .$exception->getLine() ." 行 。<br />"; 
@fclose ($fp); 

区 


程序 运行 结果 如 图 12-15 所 示 。 
@ [OB repvecaha- P - ©¢ | Bochoe 


文 全 (站 区 志和 看 WV)】 收藏 Al 工具 [超生 H) 


文件 路 径 有 误 或 找 不 到 文件 。 在 文件 C: \wamp\www\code\ch12 
\12. 12. php 的 6 行 . 


到 125% ~ 
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图 12-15 ”处 理 异常 或 错误 
【案例 剖析 】 
(1) fopen0) 函 数 打开 $SDOCUMENT ROOT/book :txt 文件 进行 读 取 ， 但 是 由 于 book.txt 文 


件 不 存在 ， 所 以 $fp 为 false。 
(2) try 区 块 判 断 Sf 为 false 时 ， 抛 出 一 个 异常 ， 此 异常 直接 通过 new 关键 字 生成 
Exception0 类 的 实例 。 异 常 信 息 是 传 入 参数 定义 的 “文件 路 径 有 误 或 找 不 到 文件 。” 
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(3) catch 区 块 通过 处 理 传 入 的 Exception0 类 实例 ， 显 示 出 错误 信息 、 错 误 文 件 、 错 误 发 
生 的 行 号 。 这 些 是 通过 直接 调用 Exception0 类 实例 Sexception 的 内 置 类 方法 获得 的 。 错 误 信 息 
由 getMessage(0 生 成 ， 错 误 文 件 由 getFile0 生 成 ， 错 误 发 生 行 号 由 getLine0 生 成 。 

(4) @felose0 和 @S$fp= fopen0 中 的 “@” 表 示 屏 蔽 此 命令 执行 中 产生 的 错误 信息 。 


12.6 疑难 解 惑 


疑问 1: 处 理 异 常 有 什么 规则 ? 


答 : 在 处 理 异 常 时 ， 有 下 列 规则 需要 用 户 牢 牢 掌握 。 

(1) 需要 进行 异常 处 理 的 代码 应 该 放 入 try 代码 块 内 ， 以 便 捕获 潜在 的 异常 。 
(2) 每 个 try 或 throw 代码 块 必须 至 少 拥 有 一 个 对 应 的 catch 代码 块 。 

(3) 使 用 多 个 catch 代码 块 可 以 捕获 不 同 种 类 的 异常 。 

(4) 可 以 在 try 代码 块 内 的 catch 代码 块 中 再 次 抛 出 Cre-thrown) 异 常 。 


疑问 2: 如 何 隐藏 错误 信息 ? 


答 : PHP 提供 了 一 种 隐藏 错误 的 方法 。 就 是 在 被 调用 的 函数 名 前 加 @ 符 号 ， 这 样 会 隐藏 
可 能 由 于 这 个 函数 导致 的 错误 信息 。 





例如 以 下 代码 : 

<?php 
sab = Fopen(m 1l238 txt men) // 打 开 指 定 的 文件 
fclose(); // 关 闭 指定 的 文件 

时 

由 于 指定 的 文件 不 存在 ， 所 以 运行 后 会 弹出 如 图 12-16 所 示 的 错误 信息 。 


a x 
@ HE Bea 5- 6 | Ba @ 
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Callstack 2 
# [Time [Memory Function [Location 
1| 0.0009 351264| (nain] () Vi. 13. Php:0 





[2] ooo | 351313iapen () |..\2.13.php:2 


12.13.php:0 
| 12. 13. php:3 “| 














12-16 ”出 现 错误 信息 


如 果 在 fopen0 函 数 和 fclose0 函 数 前 加 上 @ 符 号 ， 再 次 运行 程序 时 ， 就 不 会 出 现 错误 信息 
了 。 这 种 隐藏 信息 的 方法 对 于 查找 错误 的 位 置 是 很 有 帮助 的 。 
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13 章 
与 外 界 的 交流 
操作 文件 
与 目录 





在 前 面 的 表单 章节 中 ， 已 经 实现 了 用 form 发 送 数据 给 PHP，PHP 再 处 理 数 据 
并 输出 HTML 给 浏览 器 。 在 这 样 的 一 个 流程 里 ， 数 据 会 直接 被 PHP 代码 处 理 成 
HITML。 如 果 想 把 数据 储存 起 来 ， 并 在 需要 的 时 候 读 取 和 处 理 ， 该 怎么 办 呢 ? 这 就 
是 本 章 需 要 解决 的 问题 。 在 使 用 PHP 开发 网 站 的 过 程 中 ， 文 件 的 操作 大 致 分 为 对 
普通 文本 的 操作 和 对 数据 库 文件 的 操作 。 本 章 主 要 讲述 如 何 对 普通 文件 进行 写 入 和 
读 取 ， 以 及 目录 的 操作 、 文 件 的 上 传 等 操作 。 
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13.1 查看 文件 和 目录 
在 掌握 PHP 操作 文件 和 目录 的 技能 前 ， 首 先 需要 学 习 如 何 查看 文件 和 目录 。 


13.1.1 查看 文件 名 称 


使 用 basename() 函 数 可 以 查看 文件 的 名 称 ， 该 函数 返回 文件 目录 中 去 掉 路 径 后 的 文件 名 
称 。 其 语法 格式 如 下 : 


basename ($path, $suffix) 


其 中 参数 $path 为 必需 参数 ， 指 定 要 检查 的 路 径 ， 参 数 $suffix 为 可 选 参数 ， 规 定 文件 的 扩 
展 名 。 如 果 文 件 有 $suffix， 则 不 会 输出 这 个 扩展 名 。 
【 例 13.1】 查 看 文件 名 称 (示例 文件 ch13\13.1.php)。 














<?php 
$path = "/test/index.php"; 
// 显 示 带 有 文件 扩展 名 的 文件 名 OO 
echo basename ($path)."<br/> "; Er 
index. ph 
// 显 示 不 带 有 文件 扩展 名 的 文件 名 de 
echo basename ($path,".php"); 
> 所 125% 
运行 13.1.php 文件 ， 结 果 如 图 13-1 所 示 。 图 13-1 查看 文件 名 称 


Se basename0 〇 函数 只 查看 $path 变量 中 的 文件 名 称 ， 并 不 核实 该 目录 是 否 真实 存在 。 


13.1.2 ”查看 目录 名 称 


使 用 dimameO 函 数 可 以 查看 目录 的 名 称 ， 该 函数 返回 文件 目录 中 去 掉 文 件 后 的 目录 名 
称 。 其 语法 格式 如 下 : 


dirname ($path) 
其 中 参数 $path 为 必需 参数 ， 指 定 要 检查 的 路 径 。 
【 例 13.2】 查 看 路 径 名 称 (示例 文件 ch13\13.2.php)。 








<?php | 人 

$path = "/web/test/index.php"; -eeehe. Pd | 四 Ce 
// 显 示 路 径 的 名 称 ES ND EM WA IRT Wa 

echo dirname ($path); /web/test 

> 

运行 13.2.php 文件 ， 结 果 如 图 13-2 所 示 。 | aa 





13-2 ”查看 路 径 的 名 称 


人 dimmameO 函 数 只 查看 $path 变量 中 的 目录 名 称 ， 并 不 核实 该 目录 是 否 真实 存在 。 
13.1.3 ”查看 文件 真实 目录 


使 用 readpathO 可 以 查看 文件 的 真实 目录 ， 该 函数 返回 绝对 路 径 。 它 会 删除 所 有 符号 连接 
(比如 //, WY. 以 及 多 余 的 ")， 返 回绝 对 路 径 名 称 。 语 法 格式 如 下 : 


淹 四 J 在 装 认 灌 一 一 滞 叶 下 浊 六 上 机 8 小 全 


realpath ($path) 
其 中 参数 $path 为 必需 参数 ， 指 定 要 检查 的 路 径 。 如 果 
文件 不 存在 ， 则 返回 false。 本 
【 例 13.3】 查 看 真实 路 径 ( 示 例文 件 ch13\13.3.php)。 |¢ [Ocnr ry) TT 
交 件 站” 编 蝇 (E 二 看 (V) 收 宫 夫 (A) 工具 (T) 才 助 IH) 














<?php C: \wamp\www \code\ch13\13. 3. php 


$path = "13.3.php"; SN 
// 显 示 绝 对 路 径 \ 
echo realpath ($path); 

R125% ~ 


?> 
图 13-3 ”查看 真实 路 径 





运行 13.3.php 文件 ， 结 果 如 图 13-3 所 示 。 


13.2 ”查看 文件 信息 


PHP 提供 了 查看 文件 基本 信息 的 函数 。 通 过 这 些 函 数 ， 用 户 可 以 查看 文件 的 类 型 、 查 看 
文件 的 访问 和 修改 时 间 、 获 取 文 件 的 权限 等 。 


13.2.1 查看 文件 的 类 型 


使 用 filetypeO 函 数 可 以 获取 文件 的 类 型 。 可 能 返回 值 有 fifo、char、dir、block、link、file 
和 unknown。 语 法 格式 如 下 : 

filetype ($filename) 

其 中 参数 $filename 为 必需 参数 ， 指 定 要 检查 的 文件 路 径 。 如 果 查 看 失败 ， 则 返回 false。 

【 例 13.4】 查 看 文件 的 类 型 (示例 文件 ch13\13.4.php)。 





<?php 
$path = "D:/test/"; @s 苞 es 
echo filetype ($path) ."<br/> ";// 显 示 文件 的 类 型 为 dir A 


dir 
$pathl = "13.4.php"; file 
echo filetype ($path1) ;// 显 示 文件 的 类 型 为 file os 


2> 


运行 13.4.php 文件 ， 结 果 如 图 13-4 所 示 。 





13-4 ”查看 文件 的 类 型 
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13.2.2 ”查看 文件 的 访问 和 修改 时 间 
使 用 fileatimeO 函 数 可 以 获取 文件 上 次 的 访问 时 间 。 语 法 格式 如 下 


fileatime ($filename) 


其 中 参数 $filename 为 必需 参数 ， 指 定 要 检查 的 文件 名 称 。 如 果 查 看 失败 ， 则 返回 false。 
【 例 13.5】 查 看 文件 的 访问 时 间 ( 示 例文 件 ch13\13.5.php)。 


<?php 








$path = "T1301.php"y 

echo fileatime ($path)."<br/> "; // 显 示 文 件 上 次 的 访问 时 间 

echo date("Y-m-d H:i:s ",fileatime ($path));// 设 置 时 间 的 显示 格式 

区 光 

运行 13.5.php 文件 ， 结 果 如 图 13-5 所 示 。 从 结果 可 以 看 pe x 
出 ， 在 默认 情况 下 返回 的 时 间 以 Unix 时 间 鹤 的 形式 。 A 

使 用 filemtimeO 函 数 可 以 获取 文件 上 次 被 修改 的 时 间 。 语 20938830 10.14:36 
法 格式 如 下 : 


R25% 





filemtime ($filename) 


其 中 参数 $filename 为 必需 参数 ， 指 定 要 检查 的 文件 名 
称 。 如 果 查 看 失败 ， 则 返回 false。 
【 例 13.6】 查 看 文件 的 修改 时 间 ( 示 例文 件 ch13\13.6.php)。 


13-5 查看 文件 的 访问 时 间 


<?php - OO x 
spath = "YT3 .3 PhD © |B r/ocaho.. 只- © | Biccahon 
echo filemtime ($path)."<br/> "; JR 
es 5023332: 

// 显 示 文件 上 次 的 修改 时 间 2017-08210 10:44:28 
echo date("Y-m-d H:i:s ",fileatime ($path)); 

// 设 置 时 间 的 显示 格式 ls% 
?> 





本 图 13-6 查看 文件 的 修改 时 间 
运行 13.6.php 文件 ， 结 果 如 图 13-6 所 示 。 


13.3 文件 操作 


在 不 使 用 数据 库 系统 的 情况 下 ， 数 据 可 以 通过 文件 (File) 来 实现 数据 的 储存 和 读 取 。 这 个 
数据 存 取 的 过 程 也 是 PHP 处 理 文件 的 过 程 。 这 里 涉及 的 文件 是 文本 文件 (Text File)。 


13.3.1 打开 文件 和 关闭 文件 
对 文件 操作 前 ， 需 要 打开 文件 。PHP 提供 的 fopen0 函 数 可 以 打开 文件 。 语 法 格式 如 下 : 


fopen ($filename, $mode) 


其 中 参数 $filename 为 必需 参数 ， 指 定 要 打开 的 文件 名 称 。 参 数 Smode 为 打开 文件 的 方 
式 ， 其 取 值 如 表 13-1 所 示 。 


Dd 
Gi 


fe 


表 13-1 fopen() 函 数 中 参数 mode 的 取 值 





mode 取 值 含义 
E 打开 文件 为 只 读 。 文 件 指针 在 文件 的 开头 开始 
w 打开 文件 为 只 写 





打开 文件 为 只 写 。 文 件 中 的 现 有 数据 会 被 保留 。 文 件 指针 在 文件 结尾 开始 。 如 果 
文件 不 存在 ， 则 创建 新 的 文件 

x 创建 新 文件 为 只 写 。 如 果 文件 已 存在 ， 返 回 FALSE 和 错误 

tt 打开 文件 为 读 / 写 。 文件 指针 在 文件 开头 开始 

打开 文件 为 读 / 写 。 如 果 文 件 不 存在 ， 则 删除 文件 的 内 容 或 创建 一 个 新 的 文件 。 
文件 指针 在 文件 的 开头 开始 

打开 文件 为 读 / 写 。 文 件 中 己 有 的 数据 会 被 保留 。 文 件 指针 在 文件 结尾 开始 。 如 
果 文 件 不 存在 ， 则 创建 新 的 文件 

x+ 创建 新 文件 为 读 / 写 。 如 果 文 件 已 存在 ， 返 回 FALSE 和 错误 


文件 操作 完成 后 ， 需 要 关闭 文件 ， 从 而 释放 资源 。 关 闭 文件 使 用 fclose0 函 数 。 语 法 格式 
如 下 : 


bool fclose (resource handle) 


其 中 参数 handle 为 已 经 打开 文件 的 资源 对 象 。 如 果 handle 无 效 ， 则 返回 false。 
【 例 13.7】 打 开 和 关闭 文件 (示例 文件 ch13\13.7.php)。 








名 
济 右 由 齐 准 稚 满 一 一 兽 准 如 河池 厢 册 引 洪熙 








A 





<?php 
$file = "test.txt"; 
if(($fp = fopen($file , "wb")) === false) { 


die ("使 用 写 入 方式 打开 " .$file. "文件 失败 <br/>"); 
} else { 
echo "使 用 写 入 方式 打开 " .$file. "文件 成 功 <zbr/>"; 


; 

if(fclose ($fp)){ 

echo "文件 " .$file." 关 闭 成 功 <br/>"; 
} else { 

echo "文件 " .$file." 关 闭 失败 <br/>"; 
时 


运行 13.7.php 文件 ， 结 果 如 图 13-7 所 示 。 -x 
[a rE] em 
ET 


13.3.2” 读 取 文 件 使 用 写 入 方式 打开 test. txt 文 件 成 功 


文件 test_txt 关 闭 成 功 











打开 文件 后 ， 即 可 读 取 文 件 的 内 容 。PHP 提供 了 很 多 读 Sra 
取 文 件 的 方法 。 


1. 逐 行 读 取 文件 
fgets0 函 数 用 于 从 文件 中 逐 行 读 取 文件 。 





13-7 ”打开 和 关闭 文件 
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【 例 13.8】 逐 行 读 取 文 件 (示例 文件 ch13\13.8.php)。 


<?php 
$file = fopen("test.txt",，"r") or exit ("无 法 打开 文件 !"); 
// 读 取 文件 每 一 行 ， 直 到 文件 结尾 
while(!feof ($file)) 
3 
echo fgets($file). "<br/>"; 
} 
fclose ($file); // 关 闭 文件 


2> 


运行 13.8.php 文件 ， 结 果 如 图 13-8 所 示 。 


oO x 





后 - 要 Papy/ocaho- P - 0 | @ localhost 


四 文 习 。 病 络 (日 。 直 看 [V) 收 荐 夫 [A] 县 Tm 乔 助 (H) 
届 
【案例 剖析 】 十 角 数 校 梅 ， 法 塞 独自 开 。 


遥 知 不 是 雪 ， 为 有 了 瞳 香 来 。 


上 述 代码 会 逐 行 读 取 文 件 。 这 里 用 到 了 feofO 函数 ， 该 函 
数 的 作用 是 检查 是 否 已 经 到 了 文件 的 末尾 (EOF)。 在 循环 遍历 
未 知 长 度 的 数据 时 ，feof() 函数 很 有 用 。 


Os 在 w、a 和 x 模式 下 ， 用 户 无 法 读 取 打开 的 文件 。 


2. 逐 字符 读 取 文 件 

fgetc() 函数 用 于 从 文件 中 逐 字 符 地 读 取 文 件 。 

【 例 13.9】 逐 字符 读 取 文 件 (示例 文件 ch13\13.9.php)。 

<?php 

$file = fopen("testl.txt",，"r") or exit ("无 法 打开 文件 !"); 
// 读 取 文件 每 一 行 ， 直 到 文件 结尾 

while(!feof ($file)) 


"| 
echo fgetc($file). "<br/>"; 


} 
fclose ($file); // 关 闭 文件 
> 


运行 13.9.php 文件 ， 结 果 如 图 13-9 所 示 。 





| mpyocahoc eose /hi P ©| Biceohon 
六 和 个 豆 WM 贞 本 二 人) 工具。 尖 H 





eso mom Po 四) 








13-9 逐 字符 读 取 文 件 
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图 13-8 逐 行 读 取 文件 





3. 读 取 整个 文件 的 内 容 


如 果 想 读 取 整 个 文件 的 内 容 ， 可 以 使 用 readfile0、file0 或 fle_get_content0 中 的 任意 一 个 
函数 。 
【 例 13.10】 读 取 整 个 文件 (示例 文件 ch13\13.10.php)。 


<?php 
$file = "test.txt"; 
// 使 用 readfile() 函 数 读 取 文件 内 容 
readfile ($file); 
echo "<hr/>"; 
// 使 用 read () 函数 读 取 文件 内 容 
$farr = file($file) 7 
foreach ($farr as $v) 1{ 
echo $v."<br/>"; 


济 加 朵 证 闪 稚 满 一 一 兽 准 号 润 字 UT 坤 821 加 全 


echo "<hr/>"; 

// 使 用 file_get_content () 函数 读 取 文 件 内 容 
echo file_get_contents ($file); 

2 


运行 13.10.php 文件 ， 结 果 如 图 13-10 所 示 。 





© NE Sr | Geomen 


文 ff 将。 间 要 (W) 站 (A) 工 只。 大风 HH) 
墙角 数 枝 梅 ， 凌 塞 独自 开 。 遥 知 不 是 雪 ， 为 有 暗 香 来 。 


墙角 数 枝 梅 ， 凌 塞 独自 开 。 
运 知 不 是 雪 ， 为 有 了 瞳 香 米 。 


墙角 数 枝 梅 ， 凌 寒 独自 开 。 遥 知 不 是 雪 ， 为 有 暗 香 来 。 











125% ~ 





13-10” 读 取 整 个 文件 
13.3.3 ”文件 数据 写 入 


对 于 一 个 文件 的 “ 写 ” 操 作 ， 基 本 步骤 如 下 。 

(1) 打开 文件 。 

(2) 从 文件 里 读 取 数 据 ， 或 者 向 文件 中 写 入 数据 。 

(3) 关闭 文件 。 

打开 文件 的 前 提 是 ， 文 件 首先 是 存在 的 。 如 果 不 存在 ， 则 需要 建立 一 个 文件 。 并 且 在 所 
在 的 系统 环境 中 ， 代 码 应 该 对 文件 具有 “ 写 ” 的 权限 。 

通过 使 用 fwrite0 或 file put contents0 函 数 ， 可 以 对 文件 写 入 数据 。 

fwrite0) 函 数 的 语法 格式 如 下 : 


fwrite (file,string,length) 


其 中 file 为 必需 参数 ， 指 定 要 写 入 的 文件 。 如 果 文 件 不 存在 ， 则 创建 一 个 新 文件 。string 
为 必需 参数 ， 指 定 要 写 入 文件 的 字符 串 。length 为 可 选 参数 ， 指 定 要 写 入 的 最 大 字 节 数 。 


和 
19 售 


PHP+MySQL 动态 网 站 开发 
案例 课堂 (第 2 版 ) B… 


file put_contentsO) 函 数 的 语法 格式 如 下 : 


file_put_contents (file,data,mode,context) 


其 中 file 为 必需 参数 ， 指 定 要 写 入 的 文件 。 如 果 文 件 不 存在 ， 则 创建 一 个 新 文件 。data 为 
可 选 参数 ， 指 定 要 写 入 文件 的 数据 。 可 以 是 字符 串 、 数 组 或 数据 流 。mode 为 可 选 参数 ， 指 定 
如 何 打 开 / 写 入 文件 。context 为 可 选 参数 ， 规 定 文件 句柄 的 环境 。 

【 例 13.11】 文 件 写 入 数据 (示例 文件 ch13\13.11.php)。 


<?php 

$file = "gushi.txt"; 

$str = "过 过 牵 牛 星 ， 议 拉 河 汉 女 。 纤 纤 扬 素 手 ， 杞 村 弄 机 村 。"; 
// 使 用 fwrite () 函数 写 入 文件 

$fp = fopen($file，"wb") or die(" 打 开 文件 错误 ! ") 
fwrite($fp , $str); 

fclose ($fp); 

readfile ($file); 

echo: "<hr/>"s 





$str = " 河 汉 清 且 浅 ， 相 去 复 几 许 ! 春 春 一 水 间 ， 脉 脉 不 得 语 。"; 
// 使 用 file_put_contens () 函数 往 文件 追加 内 容 
file_put_contents ($file , $str ， FILE APPEND); 
readfile ($file); 

2 


运行 13.11.php 文件 ， 结 果 如 图 13-11 所 示 。 
打开 gushi.txt 文件 ， 可 以 查看 写 入 的 内 容 ， 如 图 13-12 所 示 。 


[ages | 
en nae mR cen Tam weet a 


通 党 素 牛 星 ， 食 由 河 汉 女 。 纤 纤 所 素 手 ， 札 杞 开机 本。 el 
通 当 达 牛 星 ， 掖 胶 河 汉 女 ， 纤 纤 摆 素 手 ， 札 札 并 机 杆 。 时 和 和 对 ee 


河 汉 清 且 线 ， 相 去 复 几 许 ! 僵硬 一 水 同 ， 脉 对 不 得 语 。 


=- o x 





















13-11 文件 写 入 数据 图 13-12 ”查看 文件 的 内 容 


13.3.4” 重 命名 和 移动 文件 


rename() 函 数 可 以 重 命名 文件 或 目录 。 若 成 功 ， 则 该 函数 返回 true; 若 失败 ， 则 返回 
false。 语 法 格式 如 下 ; 


rename (oldname, newname, context) 


其 中 oldname 为 必需 参数 ， 指 定 需要 重 命 名 文件 或 目录 ; newname 为 必需 参数 ， 指 定 文 
件 或 目录 的 新 名 称 ;，context 为 可 选 参数 ， 规 定 文件 句柄 的 环境 。 

这 里 需要 注意 的 是 : 如 果 源 文件 和 目标 文件 路 径 相同 ， 可 以 实现 文件 的 重 命 名 ; 如 果 源 
文件 和 目标 文件 路 径 不 相同 ， 可 以 实现 移动 文件 的 效果 。 

【 例 13.12】 重 命名 和 移动 文件 (示例 文件 ch13\13.12.php)。 








<?php 

$file = “test.txt"; 

Snewfile = 'newtest.txt'; 

// 文件 的 重 命名 

if (rename ($file, $newfile)) { 


echo "文件 重 命名 成 功 !"; 


} else { 

echo "文件 重 命名 失败 ! "; 
3 
// 文件 的 移动 


$movefile = "../test.txt"; 

if (rename ($newfile, $movefile)) { 
echo "文件 移动 成 功 !"; 

} else { 
echo "文件 移动 失败 ! "; 

天 


攻关 


运行 13.12.php 文件 ， 结 果 如 图 13-13 所 示 。 
13.3.5 复制 文件 
使 用 copy0 函 数 可 以 复制 文件 。 语 法 格式 如 下 : 


copy (source, destination) 


其 中 source 为 必需 参数 ， 指 定 需要 复制 的 文件 ，destination 为 必需 参数 ， 指 定 复 制 文件 的 


目的 地 。 
【 例 13.13】 复 制 文件 (示例 文件 ch13\13.13.php)。 


<?php 
$file = "testl.txt"; 
Snewfile = 'test2.txt'; 


if (copy ($file, $newfile)) { 
echo "文件 成 功 复制 为 " . $newfile; 
} else { 
echo "文件 复制 失败 ! "; 
} 


> 


运行 13.13.php 文件 ， 结 果 如 图 13-14 所 示 。 


13.3.6 ”删除 文件 
使 用 unlink 0 函数 可 以 删除 文件 。 语 法 格式 如 下 : 


unlink (filename) 


其 中 filename 为 必需 参数 ， 指 定 需要 删除 的 文件 。 
【 例 13.14】 删 除 文件 (示例 文件 ch13\13.14.php)。 
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图 13-13 重 命 名 和 移动 文件 
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图 13-14 复制 文件 
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<?php - OO x 

Sfile = “test2.txt"r | [@- @ rtp//ocaho.. P - © | @ localhost 

if(unlink ($file)) { | 0 © EV WN IRM wa 
echo "文件 " .$file." 删 除 成 功 ! "; 文件 test2. txt 删 除 成 功 ! 

Pelse 


echo "文件 " .$file." 删 除 失败 ! "; 





2 胞 125% 
运行 13.14.php 文件 ， 结 果 如 图 13-15 所 示 。 图 13-15 删除 文件 





13.4 目录 操作 


在 PHP 中 ,利用 相关 函数 可 以 实现 对 目录 的 操作 。 常 见 的 目录 操作 函数 的 使 用 方法 和 技 
巧 如 下 。 


1. string getcwd(void) 
该 函数 主要 是 获取 当前 的 工作 目录 ， 返 回 的 是 字符 串 。 下 面 举 例 说 明 此 函数 的 用 法 。 
【 例 13.15】 获 取 当 前 的 工作 目录 (示例 文件 ch13\13.5.php)。 


<?php 
$dl = getcwd(); // 获 取 当 前 路 径 
echo getcwd(); // 输 出 当前 目录 








?> 

程序 运行 结果 如 图 13-16 所 示 。 -日 x 
[@- Grp//ocahe. P - © | @ localhost 

2. array scandir(string directory[, int sorting_order]) 文件 多 (EF)】 二 看 V) 收藏) 工具 Tm 地 把 (H) 





C: \wamp\www\code\ch13 


该 函数 返回 一 个 array， 包 含有 directory 中 的 文件 和 
目录 。 如 果 directory 不 是 一 个 目录 ， 则 返回 布尔 值 
false， 并 产生 一 条 E_WARNING 级 别 的 错误 。 在 默认 情 R125% ~ 
况 下 ， 返 回 值 是 按照 字母 顺序 升序 排列 的 。 如 果 使 用 了 可 13.16 获取 当前 的 工作 目录 
选 参数 sorting_order( 设 为 )， 则 按 字母 顺序 降序 排列 。 

下 面 举例 说 明 此 函数 的 使 用 方法 。 

【 例 13.16】 使 用 scandir0 函 数 (示例 文件 ch13\13.16.php)。 

<?php 

$dir = 'C:/wamp/www/code/chl3'; // 定 义 指定 的 目录 


$filesl = scandir ($dir); // 列 出 指定 目录 中 文件 和 目录 
$files2 = scandir($dir, 1); 


print r($files1); // 输 出 指定 目录 中 的 文件 和 目录 
print r ($files2); 
Ep 


程序 运行 结果 如 图 13-17 所 示 。 
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Array ( [0] =>》- > [2] => 13.1.php [3] => 13.10.php [4] => 13.11.php [5] => 
13. 12. php [6 13. 14. php [8: 13. 15. php [9] => 13.16.php [1 
13. 2. php [11 > 13.4.php [13- 13. 5.php [14] => 13.6. php [15 
13.7. php [16] => 13.8.php [17] => 13.9.php [18] => gushi. txt [19] => test [20] = 
test. txt [21] => testl. txt ) Array ( [0] => testl.txt [1] => test.txt [2] => test [3] 
=> gushi. txt [4] => 13.9.php [5] => 13.8.php [6] => 13.7.php [7] => 13.6.php [8] => 
13.5. php [9] = . php [12] => 13.16.php [13] = 
13. 15. php [14] 13. 12. php [17] => 13.11.php [ 
=> 13. 10. php [19] 
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13-17 ”使 用 scandir() 函 数 
3. dir(sting directory) 
此 函数 模仿 面向 对 象 机 制 ， 将 指定 的 目录 名 转换 为 一 个 对 象 并 返回 。 使 用 说 明 如 下 : 


class dir { 


dirl(string directory) 
string path Ye 
resource handle 
string read(void) 
void rewind(void) 


void close(void) 


其 中 handle 属性 含义 为 目录 句柄 : path 属性 的 含义 为 打开 目录 的 路 径 ， 函数 read(void) 的 
含义 为 读 取 目录 ; 函数 rewind(void) 的 含义 为 复位 目录 ; 函数 close(void) 的 含义 为 关闭 目录 。 

下 面 举例 说 明 此 函数 的 使 用 方法 。 

【 例 13.17】 使 用 dir0 函 数 (示例 文件 ch13\13.17.php)。 


<?php 

$d2 = dir("C:/wamp/www/code/ch13"); 

echo "Handle: ".$d2->handle."<br/>\n"; 
echo "Path: ".$d2->path."<br/>\n"; 

while (false !== ($entry = $d2->read())) { 
echo $entry."<br/>\n"; 

. 

$d2->close(); 

人 > 


程序 运行 结果 如 图 13-18 所 示 。 
4. chdir(string directory) 


此 函数 将 PHP 的 当前 目录 改 为 directory。 下 面 举例 说 明 此 函数 的 使 用 方法 。 
【 例 13.18】 使 用 chdir0 函 数 (示例 文件 ch13\13.18.php)。 


<?php 

if(chdir("C:/wamp/www/code/ch13")){ 

echo "当前 目录 更 改 为 : C:/wamp/www/code/chl3"; 
Yelset 

echo "当前 目录 更 改 失 败 了 "; 

’ 


县 > 
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程序 运行 结果 如 图 13-19 所 示 。 


| 

@ Ee nea 
Aun WE aE eh IRT sao 

Handle: Resource id #3 

Path: C: /wanp/ ww/code/ch13 








13. 1.php 





- OO x 
13.4 php 
13.5. php @ 加 -| 外 hoifocalho.. P - © | 居 localhost 
13.7. php 文件 (有 蝙 名 (EF) 至 看 IV) 收 荐 夫 (A) ”工具 (0) 帮助 (H) 


13. 9. php 当前 目录 更 改 为 : C:/wamp/www/code/ch13 
ushi txt 
test 

tost txt 
testl, txt 
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图 13-18 使 用 dir 函数 图 13-19 使 用 chdir() 函 数 





5. void closedir(resource dir handle) 
此 函数 主要 是 关闭 由 dir_ handle 指定 的 目录 流 。 另 外 ， 目 录 流 必须 已 经 被 opendir0 打 开 。 
6. resource opendir(string path) 


返回 一 个 目录 句柄 。 其 中 path 为 要 打开 的 目录 路 径 。 如 果 path 不 是 一 个 合法 的 目录 或 者 
因为 权限 限制 或 文件 系统 错误 而 不 能 打开 目录 ， 返 回 FALSE 并 产生 一 个 E_WARNING 级 别 
的 PHP 错误 信息 。 如 果 不 想 输出 错误 ， 可 以 在 opendir0 前 面 加 上 @ 符 号 。 

【 例 13.19】 使 用 opendir0 函 数 (示例 文件 ch13\13.19.php)。 


<?php 

$dir = " C:/wamp/www/code/ch13/"; 

// 打 开 一 个 目录 ， 然 后 读 取 目 录 中 的 内 容 

4f (is-dir{(sair}y) 4 

if ($dh = opendir($dir)) { 

while (($file = readdir($dh)) !== false) { 
print "filename: $fi1e 2 f141etype: " 

. filetype($dir . $file) . "\n"; 


} 

closedir($dh); 

} 

} 

?> 

程序 运行 结果 如 图 13-20 所 示 。 
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filename: . : filetype: dir filename: .. : filetype: dir filename- 
13.1.php : filetype: file filename: 13.10.php : filetype: file filename: 
13. 11. php : filetype: file filename: 13.12.php ; 
13. 13. php : filetype: file filename: 13.14.php : 
13. 15. php : filetype: file filename: 13.16.php ; 
13. 17. php : filetype: file filename: 13.18.php : 
13. 19.php : filetype: file filename: 13.2.php : filerype: file filename: 
: file filename: 13.4.php : filetype: file filename: 
: 13.6.php : filetype: file filename: 
: 13.8.php ; filetype: file filename: 
: gushi. txt : filetype: file filename: 
: dir filename: test. txt : filetype: file filename: 
testl. txt : filetype: file 
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13-20 ”使 用 opendir() 函 数 


其 中 ，is_dir0 函 数 主要 是 判断 给 定 文 件 名 是 否 是 一 个 目录 ; readdir0 函 数 从 目录 句柄 中 读 
取 条 目 ; closedir0 函 数 关闭 目录 句柄 。 


7. string readdir(resource dir_ handle) 


该 函数 主要 是 返回 目录 中 下 一 个 文件 的 文件 名 。 文 件 名 以 在 文件 系统 中 的 排序 返回 。 
【 例 13.20】 使 用 readdir0) 函 数 (示例 文件 ch13\13.20.php)。 


<?php 

// 注 意 在 4.0.0-RC2 之 前 不 存在 !== 运算 符 

if ($handle = opendir("C:/wamp/www/code/ch13/")) { 
echo "Directory handle: $handle\n"; 

echo "Files:\n"; 

/* 这 是 正确 的 遍历 目录 方法 */ 

while (false !== ($file = readdir($handle))) 1{ 
echo "$file\n"; 

closedir($handle); 

} 


> 


程序 运行 结果 如 图 13-21 所 示 。 
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Directory handle: Resource id 23 Files: , .. 

13. 1. php 13. 10. php 13. 11. php 13.12. php 13.13. php 
13. 14. php 13. 15. php 13. 16. php 13. 17. php 13. 18. php 
13. 19. php 13. 2. php 13. 20. php 13. 3. php 13.4.php 
13. 5. php 13. 6. php 13. 7. php 13. 8. php 13.9.php 
Bgushi. txt test test, txt testl. txt 


图 13-21 使 用 readdir() 函 数 
在 遍历 目录 时 ， 有 的 人 经 常会 写 出 如 下 错误 的 遍历 方法 : 


/* 这 是 错误 的 遍历 目录 的 方法 */ 
while ($file = readdir($handle)) { 


echo "$file\n"; 
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13.5 上 传 文件 


在 网 络 上 ， 用 户 可 以 上 传 自己 的 文件 。 实 现 这 种 功能 的 方法 很 多 ， 用 户 把 一 个 文件 上 传 
到 服务 器 ， 需 要 在 客户 端 和 服务 器 端 建立 一 个 通道 来 传递 文件 的 字 节 流 ， 并 在 服务 器 进行 上 
传 操作 。 下 面 介绍 一 种 使 用 代码 最 少 且 容 易 理解 的 方法 。 


13.5.1 全 局 变量 $_FILES 


通过 使 用 PHP 的 全 局 变量 $_FILES， 用 户 可 以 从 客户 计算 机 向 远程 服务 器 上 传 文件 。 
全 局 变量 $_FILES 是 一 个 二 维 数组 ， 用 于 接收 上 传 文件 的 信息 ， 它 会 保存 表单 中 type 值 为 
file 的 提交 信息 ， 有 5 个 主要 列 ， 具 体 含义 如 下 。 

(1) $_FILES["file"]["name"]: 存放 上 传 文件 的 名 称 。 

(2) $_FILES["file"]["type"]: 存放 上 传 文件 的 类 型 。 

(3) $_FILES["file"]["size"]: 存放 上 传 文件 的 大 小 ， 以 字 节 为 单位 。 

(4) $_FILES["file"]["tmp_name"] : 存放 存储 在 服务 器 的 文件 的 临时 全 路 径 。 

(5) $_FILES["file"]["error"]: 存放 文件 上 传导 致 的 错误 代码 。 

在 $_FILES["file"]["tmp_name"] 中 ，/tmp 目录 是 默认 的 上 传 临时 文件 的 存放 地 点 ， 此 时 用 
户 必 须 将 文件 从 临时 目录 中 删除 或 移动 到 其 他 位 置 ， 否 则 上 传 的 文件 会 被 自动 删除 。 可 见 ， 
无 论 上 传 是 否 成 功 ， 程 序 最 后 都 会 自动 删除 临时 目录 中 的 文件 ， 所 以 在 删除 前 ， 需 要 将 上 传 
的 文件 复制 到 其 他 位 置 ， 这 样 才 算 真 正 完 成 了 上 传 文件 的 过 程 。 

另外 ，$_FILES["file"]["error"] 中 返回 的 错误 代码 的 常量 对 应 的 数值 的 含义 如 下 。 

(1) UPLOAD ERR OK=0: 表示 没有 发 生 任何 错误 。 

(2) UPLOAD ERR INI SIZE=1: 表示 上 传 文件 的 大 小 超过 了 约定 值 。 

(3) UPLOAD ERR FORM SIZE =2: 表示 上 传 文件 的 大 小 超过 了 HTML 表单 隐藏 域 属性 
的 MAX _FILE_SIZE 元 素 所 规定 的 最 大 值 。 

(4) UPLOAD_ERR_PARTIAL =3: 表示 文件 只 被 部 分 上 传 。 

(5) UPLOAD_ERR_ NO_FILE =4: 表示 没有 上 传 任何 文件 。 


13.5.2 文件 上 传 
在 PHP 中 ， 使 用 move_uploaded_file0 函 数 可 以 将 上 传 的 文件 移动 到 新 位 置 。 语 法 格式 如 下 : 


move_uploaded file(file,newloc) 


其 中 file 为 需要 移动 的 文件 ，newloc 参数 为 文件 的 新 位 置 。 如 果 file 指定 的 上 传 文件 是 
合法 的 ， 则 文件 被 移动 到 newloc 指定 的 位 置 ， 如 果 file 指定 的 上 传 文件 不 合法 ， 则 不 会 出 现 
任何 操作 ，move_uploaded file0 函 数 将 返回 false; 如 果 file 指定 的 上 传 文件 是 合法 的 ， 但 出 
于 某 些 原 因 无 法 移动 ， 不 会 出 现任 何 操作 ，move_uploaded file0 函 数 将 返回 false， 此 外 还 会 
发 出 一 条 警告 。 








yy 


@ move_uploaded fileO 函 数 只 能 用 于 通过 HTTP POST 上 传 的 文件 。 如 果 目 标 文件 
hn 已 经 存在 ， 将 会 被 覆盖 。 


下 面 通 过 案例 来 学 习 上 传 图 片 文件 的 方法 和 技巧 。 
【 例 13.21】 上 传 图 片 文件 (示例 文件 ch13\13.21.html 和 13.21.php)。 
首先 创建 一 个 获取 上 传 文件 的 页 面 ， 文 件 名 为 13.21.html。 代 码 如 下 : 


<!DOCTYPE html> 

<html> 

<head> 

<title> 上 传 图 片 文 件 </title> 

</head> 

<body> 

<form action="13.21.php" method="post" enctype="multipart/form-data"> 
<label for="file"> 文 件 名 : </label> 
<input type="file" name="file" id="file"><br/> 
<input type="submit"”name=" submjit"” value=" 上 传 "> 

</form> 

</body> 

</html> 
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其 中 <form action="13.21.php"” method="post” enctype="multipart/form-data"> 语 句 中 的 
method 属性 表示 提交 信息 的 方式 是 post， 即 采用 数据 块 ，action 属性 表示 处 理 信息 的 页 面 为 
13.21.php; enctype="multipart/form-data" 表 示 以 二 进 制 的 方式 传递 提交 的 数据 。 

EEJRP 接着 创建 一 个 实现 文件 上 传 功能 的 文件 。 为 了 设置 和 保存 上 传 文件 的 路 径 ， 用 

户 需要 在 创建 文件 的 目录 下 新 建 一 个 名 称 为 “upload ”的 文件 夹 。 然 后 新 建 
13.21.php 文件 。 代 码 如 下 : 

<?php 

// 允许 上 传 的 图 片 后 缀 

$allowedExts = array ("gif", "jpeg", "jpg", "png"); 

$temp = explode(".", $_FILES["file"] ["name"]); 


echo $_FILES["file"] ["size"]; 
$extension = end($temp); // 获取 文件 后 级 名 





if ((($_FILES["file"] ["type"] == "image/gif") 
11 ($_FILES["file"]["type"] == "image/jpeg") 
Ml (SFILES["file"] ["type”"] == "image/ijpg") 

11 ($_FILES["file"]["type"] == "image/pjpeg") 
11 ($_FILES["file"] ["type"] == "image/x-png") 
11 ($_FILES["file"]["type"] == "image/png")) 


&& ($_FILES["file"]["size"] < 204800) // 小 于 200 kb 
&& in array ($extension, $allowedExts)) 


4 SEILES["FTLe"TI"error"l > 0 

{ 
echo "错误 ; : " .$FILES["file"] ["error"] 。 “<br/>"; 

} 

els 

' 
echo "上 传 文件 名 : " . $_FILES["file"]["name"] . "<br/>"; 
echo "文件 类 型 : " - $_FILES["file"]["type"] . "<br/>"; 
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echo "文件 大 小 : " . ($_FILES["file"] ["size"] / 1024) . " kB<br/>"; 
echo "文件 临时 存储 的 位 置 : " . $_FILES["file"]["tmp name"] . "<br/>"; 


// 判断 当期 目录 下 的 upload 目录 是 否 存在 该 文件 

// 如 果 没 有 upload 目录 ， 你 需要 创建 它 ，upload 目录 权限 为 777 

if (file exists("upload/" . $_FILES["file"] ["name"])) 
echo $_FILES["file"]["name"] - " 文件 已 经 存在 。 "; 

} 


else 





// 如 果 upload 目录 不 存在 该 文件 则 将 文件 上 传 到 upload 目录 下 
move_uploaded file($_FILES["file"] ["tmp_name"]，"upload/" . 
$_FILES["file"] ["name"]) 7 
echo "文件 存储 在 : " . "upload/"” . $_FILES["file"] ["name"]7 
)， 
} 
i 


Sse 


echo "非法 的 文件 格式 "; 
加 
运行 13.21.html 网 页 ， 结 果 如 图 13-22 所 示 。 单 击 “ 浏 览 ”按钮 ， 即 可 选择 需要 上 传 的 文 
件 ， 最 后 单 击 “ 上 传 ” 按 钮 ， 即 可 跳 转 到 13.21.php 文件 ， 如 图 13-23 所 示 ， 实 现 了 文件 的 上 


























传 操作 。 
二 
a | € [cnr Tar ET Y 
[®- @ tpi/ocalho-. P - | 居 上 Ek 件 AN WE) SEV MA IRT BNO 
Er NA) IRT SIH) 
NE SIEM 文件 名 称 : 123. jpg 
文件 名 : 浏览 . 文件 类 型 : image/ jpeg 
上 传 文 26. 3466796875 kB 
临时 文 忻 路 径 : C: \wamp\tmp\phpFA64. tmp 
123. jpg already exists. 
戈 125% 起 125% > 








13-22 ”上传 文件 图 13-23 上传 文件 的 信息 


13.6 ”案例 实战 一 一 编写 访客 计数 器 


下 面 通过 对 文本 文件 的 操作 ， 利 用 相关 函数 编写 一 个 简单 的 文本 类 型 的 访客 计数 器 。 
【 例 13.22】 编 写 访客 计数 器 (示例 文件 ch13\13.22.php)。 


<!DOCTYPE html> 

<html> 

<head> 

<title> 访 客 计数 器 </title> 

</head> 

<body> 

<?php 

if (!1@$fp=fopen("coun.txt","r")){ // 以 只 读 方 式 打 开 coun.txt 文件 


合 202 


echo "coun .txt 文件 创建 成 功 ! <br/>"; 

1 

@$num = fgets ($fp,12); // 读 取 11 位 数字 

if ($num=="") $num=0; // 如 果 文 件 的 内 容 为 空 ， 初 始 化 为 0 


$num++; // 浏 览 次 数 加 1 

@fclose ($fp); // 关 闭 文件 

$fp = fopen("coun.txt"，"w"); // 以 只 写 方式 打开 coun.txt 文件 
fwrite ($fp, $num); // 写 入 加 1 后 的 结果 

fclose ($fp); // 关 闭 文件 

echo "您 是 第 " . $num. "位 浏览 者 !"; ”// 浏 览 器 输出 浏览 次 数 

人 

</body> 

</html> 


程序 第 一 次 运行 时 ， 结 果 如 图 13-24 所 示 。 多 次 刷新 页 面 后 ， 即 可 看 到 数字 发 生 了 变 
化 ， 如 图 13-25 所 示 。 


浏 四 Jr 在 装 认 灌 一 一 滞 叶 下 浊 六 上 机 8 小 鳃 




















到 口 x — ks 口 x 
@ -pocaho.. P - ©| GE x @ [®- Sreecaho.. P+ © | GB is 
| 文件 (月 ” 妨 狠 FE) 二 看 V) 。 必 荫 夫 (A) 工具 (T) 才 动 H) 文件 F) 。 妨 辐 (E) 查看 (V| 。 收 写实 (A) 工具 (T] 部 盈 (H) 
coun. txt 文 件 创建 成 功 ! 您 是 第 4 位 浏览 者 ! 
您 是 第 1 位 浏览 者 ! 
本 125% > | 
图 13-24 ”程序 第 一 次 运行 的 结果 图 13-25 多 次 刷新 页 面 后 的 结果 





由 结果 可 以 看 出 ， 该 程序 首先 创建 了 一 个 count.txt 文本 文件 ， 用 于 保存 浏览 次 数 ， 接 
着 ， 打 开 这 个 文件 ， 将 数据 初始 化 为 0， 并 实现 加 1 操作 。 


13.7 疑难 解 惑 


疑问 1: 如 何 批量 上 传 多 个 文件 ? 

答 : 本 章 讲 述 了 如 何 上 传单 个 文件 ， 那 么 如 何 上 传 多 个 文件 呢 ? 用 户 只 需要 在 表单 中 使 
用 与 复 选 框 相同 的 数组 式 提交 语法 即 可 。 

提交 的 表单 语句 如 下 : 


<form method="post" action="13.5.php" enctype="multipart/form-data"> 
<table border=0 cellspacing=0 cellpadding=0 align=center width="100%"> 
EL 

<td> 

<input name="userfile[]" type="file™ value=" 浏 览 Ta VS 

<input name="userfile[]"”type="file" value=" 浏 览 ps 

<input name="userfile[]" type="file" value=" 浏 览 3" /> 

<input type="submit" value=" 上 传 " name="B1l" /> 

</td> 

/Ee 

</table> 

</form> 


和 
203@ 
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疑问 2: 如 何 从 文件 中 读 取 一 行 ? 

答 : 在 PHP 网 站 开发 中 ， 支 持 从 文件 指针 中 读 取 一 行 。 使 用 string fgets(int handle[,int 
lengtb]) 函 数 即 可 实现 上 述 功能 。 其 中 int handle 是 要 读 入 数据 的 文件 流 指 针 ， 由 fopen0 函 数 
返回 数值 ，int length 设置 读 取 的 字符 个 数 ， 读 入 的 字符 个 数 为 length-1。 如 果 没 有 指定 
length， 则 默认 为 1024 个 字 节 。 

疑问 3: 如 何 获取 文件 或 目录 的 权限 ? 

答 : PHP 提供 的 fileperms0 函 数 可 以 返回 文件 或 目录 的 权限 。 如 果 成 功 ， 则 返回 文件 的 
访问 权限 。 如 果 失 败 ， 则 返回 false。 语 法 格式 如 下 : 


fileperms ($filename) 


其 中 参数 $filename 为 需要 检查 的 文件 名 。 





14 章 
图 形 界面 设计 
一 一 GD 绘图 与 

图 像 处 理 


PHP 不 仅 可 以 输出 纯 HTML， 还 可 以 创建 及 操作 多 种 不 同 图 像 格 式 的 图 像 文 
件 ， 包括 GIF、PNG、JPG、WBMP、XPM 等 。 更 方便 的 是 ，PHP 可 以 直接 将 图 像 
流 输 出 到 浏览 器 。 要 处 理 图 像 ， 需 要 在 编译 PHP 时 加 上 图 像 函 数 的 GD 库 '， 另 
外 ， 还 可 以 使 用 第 三 方 的 图 形 库 。 本 章 介 绍 图 形 图 像 的 处 理 方法 和 技巧 。 
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14.1 在 PHP 中 加 载 GD 库 


PHP 中 的 图 形 图 像 处 理 功 能 都 要 求 有 一 个 库 文件 的 支持 ， 这 就 是 GD2 库 。PHP 7 中 自 带 
此 库 。 

如 果 是 在 Windows 10 系统 环境 下 ， 则 修改 php.ini 中 extension=php_gd2.dll 前 面 的 “:;” 
即 可 启用 ， 如 图 14-1 所 示 。 


司 phpini -记事 本 - DO x 









map. dll 
hp_interbase. dl1 
>_ldap. dl1l 


extensi mbstring. dll 
extensj ip_exif. dll ; Must be after mbstring as it depends on it 
extensi ;_mysqli. dll 

Bxten hp_cdbc. dll 


extension=php_openssl. dll 
iextenaion=php_pdo_firobird. ll 
extension=php_pdo_mysql. dll v 








14-1 修改 php.ini 配置 文件 
下 面 了 解 一 下 PHP 中 常用 的 图 像 函 数 的 功能 ， 具 体 如 表 14-1 所 示 。 
表 14-1 图 像 函数 的 功能 

















函 数 功 能 
gd_info 取得 当前 安装 的 GD 库 的 信息 
getimagesize 取得 图 像 大 小 
a . 取得 getimagesize、exif read_data、exif thumbnail、exif imagetype 所 返回 
image_type_to_mime type 的 图 像 类 型 的 MIME 类 型 
image2wbmp 以 WBMP 格式 将 图 像 输出 到 浏览 器 或 文件 
imagealphablending 设 定 图 像 的 混 色 模式 
imageantialias 是 否 使 用 antialias 功能 
imagearc 画 椭圆 弧 
imagechar 水 平地 画 一 个 字符 
imagecharup 垂直 地 画 一 个 字符 
imagecolorallocate 为 一 幅 图 像 分 配 颜 色 
imagecolorallocatealpha 为 一 幅 图 像 分 配 颜色 和 透明 度 
imagecolorat 取得 某 像素 的 颜色 索引 值 









































续 表 

函 数 功 能 
imagecolorclosest 取得 与 指定 的 颜色 最 接近 的 颜色 的 索引 值 
imagecolorclosestalpha 取得 与 指定 的 颜色 加 透明 度 最 接近 的 颜色 的 索引 值 
imagecolorclosesthwb 取得 与 给 定 颜色 最 接近 的 色 度 的 黑白 色 的 索引 
imagecolordeallocate 取消 图 像 颜色 的 分 配 
imagecolorexact 取得 指定 颜色 的 索引 值 
imagecolorexactalpha 取得 指定 的 颜色 加 透明 度 的 索引 值 
imagecolormatch 使 一 个 图 像 中 调 色 板 版 本 的 颜色 与 真 彩色 版 本 更 能 匹配 
imagecolorresolve 取得 指定 颜色 的 索引 值 或 有 可 能 得 到 的 最 接近 的 替代 值 
imagecolorresolvealpha 取得 指定 颜色 加 透明 度 的 索引 值 或 有 可 能 得 到 的 最 接近 的 替代 值 
imagecolorset 给 指定 调 色 板 索引 设 定 颜色 
imagecolorsforindex 取得 某 索引 的 颜色 
imagecolorstotal 取得 一 幅 图 像 的 调 色 板 中 颜色 的 数目 
imagecolortransparent 将 某 个 颜色 定义 为 透明 色 
imagecopy 拷贝 图 像 的 一 部 分 
imagecopymerge 拷贝 并 合并 图 像 的 一 部 分 
imagecopymergegray 用 灰 度 拷贝 并 合并 图 像 的 一 部 分 
imagecopyresampled 重 采样 拷贝 部 分 图 像 并 调整 大 小 
imagecopyresized 拷贝 部 分 图 像 并 调整 大 小 
imagecreate 新 建 一 个 基于 调 色 板 的 图 像 
imagecreatefromgd2 从 GD2 文件 或 URL 新 建 一 图 像 
imagecreatefromgd2part 从 给 定 的 GD2 文件 或 URL 中 的 部 分 新 建 一 图 像 
imagecreatefromgd 从 GD 文件 或 URL 新 建 一 图 像 
imagecreatefromgif 从 GIF 文件 或 URL 新 建 一 图 像 
imagecreatefromjpeg 从 JPEG 文件 或 URL 新 建 一 图 像 
imagecreatefrompng 从 PNG 文件 或 URL 新 建 一 图 像 
imagecreatefromstring 从 字符 串 中 的 图 像 流 新 建 一 图 像 
imagecreatefromwbmp 从 WBMP 文件 或 URL 新建 一 图 像 
imagecreatefromxbm 从 XBM 文件 或 URL 新 建 一 图 像 
imagecreatefromxpm 从 XPM 文件 或 URL 新 建 一 图 像 
imagecreatetruecolor 新 建 一 个 真 彩色 图 像 
imagedashedline 画 一 虚线 


加 
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续 表 

函 数 功 能 
imagedestroy 销毁 一 图 像 
imageellipse 画 一 个 椭 
imagefill 区 域 填 充 
imagefilledarc 画 一 椭圆 弧 且 填充 
imagefilledellipse 画 一 椭圆 并 填充 
imagefilledpolygon 画 一 多 边 形 并 填充 
imagefilledrectangle 画 一 矩形 并 填充 
imagefilltoborder 区 域 填充 到 指定 颜色 的 边界 为 止 
imagefontheight 取得 字体 高 度 
imagefontwidth 取得 字体 宽度 
imageftbbox 取得 使 用 了 FreeType 2 字体 的 文本 的 范围 
imagefttext 使 用 FreeType 2 字体 将 文本 写 入 图 像 
imagegd 将 GD 图 像 输出 到 浏览 器 或 文件 
imagegif 以 GIF 格式 将 图 像 输出 到 浏览 器 或 文件 
imagejpeg 以 JPEG 格式 将 图 像 输出 到 浏览 器 或 文件 
imageline 画 一 条 直线 
imagepng 将 调 色 板 从 一 幅 图 像 拷贝 到 另 一 幅 
imagepolygon 画 一 个 多 边 形 
imagerectangle 画 一 个 矩形 
imagerotate 用 给 定 角度 旋转 图 像 
imagesetstyle 设 定 画 线 的 风格 
imagesetthickness 设 定 画 线 的 宽度 
imagesx 取得 图 像 宽度 
imagesy 取得 图 像 高 度 
imagetruecolortopalette 将 真 彩色 图 像 转 换 为 调 色 板 图 像 
imagettfbbox 取得 使 用 TrueType 字体 的 文本 的 范围 
imagettftext 用 TrueType 字体 向 图 像 写 入 文本 


14.2 ”图 形 图 像 的 典型 应 用 案例 


下 面 讲述 图 形 图 像 的 经 典 使 用 案例 。 


二 
伟 208 


14.2.1 创建 一 个 简单 的 图 像 


使 用 GD2 库 文 件 ， 就 像 使 用 其 他 库 文 件 一 样 。 由 于 它 是 PHP 的 内 置 库 文 件 ， 不 需要 在 
PHP 文件 中 再 用 include 等 函数 进行 调用 。 下 面 通过 示例 介绍 图 像 的 创建 方法 。 
【 例 14.1】 创 建 图 像 (示例 文件 ch14\14.1.php)。 

















<?php 

$im = imagecreate(200,300); / /创建 一 个 画布 

$white = imagecolorallocate ($im，8,2,133);  ”// 设 置 画 布 的 背景 色 为 一 种 蓝 色 
imagegif ($im); // 输 出 图 

> 


运行 程序 ， 结 果 如 图 14-2 所 示 。 本 例 使 用 imagecreate0 函 数 创建 了 一 个 宽 200 像素 、 高 
300 像素 的 画布 ， 并 设置 画布 的 RGB 值 为 (8, 2, 133)， 最 后 输出 一 个 GIF 格式 的 图 像 。 








二 
Er i i ed si 
图 14-2 图 像 的 创建 
) 提 示 使 用 imagecreate(200.300) 函 数 创建 基于 普通 调 色 板 的 画布 ， 支 持 256 色 ， 其 中 


装 、 ”200、300 为 图 像 的 宽度 和 高 度 ， 单 位 为 像素 。 


上 面 的 案例 只 是 把 图 片 输出 到 页 面 ， 那 么 如 果 需 要 保存 图 片 文 件 呢 ? 下 面 通过 例子 来 介 
绍 图 像 文件 的 创建 方法 。 
【 例 14.2】 创 建 图 像 文 件 (示例 文件 ch14\14.2.php)。 


<?php 

$ysize = 200; 

$xsize = 300; 

$theimage = imagecreatetruecolor ($xsize, $ysize); 
$color2 = imagecolorallocate($theimage, 8,2,133) 
$color3 = imagecolorallocate($theimage, 230,22 
imagefill ($theimage, 0, 0, $color2); 

imagearc ($theimage,100,100,150,200,0,270,$color3); 
imagejpeg ($theimage, "newimage.jpeg"); 





); 
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header ('content-type: image/png'); 
imagepng ($theimage); 

imagedestroy ($theimage); 

2 


运行 程序 ， 结 果 如 图 14-3 所 示 。 同 时 ， 在 程序 文件 夹 下 生成 了 一 个 名 为 newimage.jpeg 
的 图 片 ， 其 内 容 与 页 面 显示 的 相同 。 
@ 由 -| 夺 hapy/localho. P76 厄 localhost 


文件 (R) 编 加 (EF) 查看 (V) 收藏 夫 (A) 工具 (T) 帮助 (H) 

















- OO x 











下 125% ~ 








图 14-3 ”页 面 效果 
【案例 剖析 】 

(1) imagecreatetruecolor0 函 数 是 用 来 创建 图 片 画布 的 。 它 需要 两 个 参数 ， 一 个 是 x 轴 的 
大 小 ， 一 个 是 y 轴 的 大 小 。$xsize=200、Sysize=300 分 别 设 定 了 这 两 参数 的 大 小 。$theimage= 
imagecreatetruecolor($xsize, $ysize) 使 用 这 两 个 参数 生成 了 画布 ， 并 且 赋 值 为 Stheimage。 

(2) imagearc($theimage,100,100,150,200,0,270，$color3) 语 句 使 用 imagearc0O) 函 数 在 画布 上 
创建 了 一 个 弧 线 。 它 的 参数 分 为 以 下 几 个 部 分 : $theimage 为 目标 画布 ，“100,100” 为 弧 线 中 
心 点 的 x、y 华 标 ，“150,200” 为 弧 线 的 宽度 和 高 度 ，“0,270” 为 顺 时 针 画 弧 线 的 起 始 度数 
和 终点 度数 ， 是 在 0 到 360 度 之 间 ，$color3 为 画 弧 线 所 使 用 的 颜色 。 

(3) imagejpeg0 函数 是 生成 JPEG 格式 的 图 片 的 函数 。 这 里 ，imagejpeg($theimage， 
“newimage.jpeg”) 把 画布 对 象 $theimage 及 其 所 有 操作 生成 为 一 个 名 为 newimage.jpeg 的 JPEG 
图 片 文件 ， 并 且 直 接 储存 在 当前 路 径 下 。 

(4) 同时 ，header('content-type: image/png') 和 imagepng($theimage) 向 页 面 输出 了 一 张 PNG 
格式 的 图 片 。 

(5) 最 后 清除 对 象 ， 释 放 资 源 。 


14.2.2 ”使 用 GD2 的 函数 在 图 片上 添加 文字 


上 面 是 如 何 创建 一 个 图 片 。 如 果 想 在 图 片上 添加 文字 ， 就 需要 修改 一 个 图 片 ， 具 体 的 过 
程 为 : 先 读 取 一 个 图 片 ， 然 后 修改 这 个 图 片 。 
【 例 14.3】 在 图 片上 添加 文字 (示例 文件 ch14\14.3.php)。 


<?php 
$theimage = imagecreatefromjpeg('newimage.jpeg'); 














$colorl = imagecolorallocate ($theimage，255,255,255)7 

$color3 = imagecolorallocate ($theimage, 230,22,22); 

imagestring ($theimage,5,60,100,'Text added to this image.',$colorl); 
header ('content-type: image/png'); 

imagepng ($theimage); 

imagepng ($theimage, 'textimage.png'); 

imagedestroy ($theimage); 

> 


运行 程序 ， 结 果 如 图 14-4 所 示 。 同 时 在 程序 所 在 的 文件 夹 下 生成 了 名 为 newimage.jpeg 
的 图 片 文件 ， 其 内 容 与 页 面 显示 相同 。 





= 让 区 


| 
@ 国 -| 夺 rapyiocaho- P - C|| 大 localhost 


文件 。 篇 晶 (E) ”查看 (V) 收藏 次 (A) 工具 (T) 帮助 (H) 





Text added to this image。 











图 14-4 在 图 片上 添加 文字 
【案例 剖析 】 


(1) imagecreatefromjpeg(mewimage.jpeg') 语 句 中 imagecreatefromjpeg() 函 数 从 当前 路 径 下 
读 取 newimage.jpeg 图 形 文件 ， 并 且 传 递 给 Stheimage 变量 作为 对 象 ， 以 待 操作 。 

(2) 选取 颜色 后 。imagestring($theimage,5,60,100,'Text added to this image.,$colorl) 语 句 中 
的 imagestring0 函 数 向 对 象 图 片 添 加 字符 串 'Text added to this image.'。 这 里 面 的 参数 中 ， 
S$theimage 为 对 象 图 片 ，“5” 为 字体 类 型 ， 这 个 字体 类 型 的 参数 从 1 一 5 代表 不 同 的 字体 ; 

“60,100” 为 字符 串 添加 的 起 始 x、y 坐标 ;'Text added to this image.' 为 要 添加 的 字符 串 ， 当 前 

只 支持 ASC 字符 ，S$colorl 为 文字 的 颜色 。 

(3) header('content-type: image/png') 和 imagepng($theimage) 语 句 共 同 处 理 了 输出 到 页 面 的 
PNG 图 片 。 之 后 ，imagepng($theimage,'textimage.png') 语 句 就 创建 文件 名 为 textimage.png 的 
PNG 图 片 ， 并 保存 在 当前 路 径 下 。 


14.2.3 ”使 用 TrueType 字体 处 理 中 文生 成 图 片 


字体 处 理 在 很 大 程度 上 是 PHP 图 形 处 理 经 常 要 面 对 的 问题 。imagestring0 函 数 默认 的 字体 
是 十 分 有 限 的 。 这 就 要 进入 字体 库 文件 。 而 TrueType 字体 是 字体 中 极为 常用 的 格式 。 例 如 ， 
在 Windows 下 打开 C:\\WINDOWS\Fonts 目录 ， 会 出 现 很 多 字体 文件 ， 其 中 绝 大 部 分 是 
TrueType 字体 ， 如 图 14-5 所 示 。 
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图 14-5 系统 中 的 字体 


PHP 使 用 GD2 库 ， 在 Windows 环境 下 ， 需 要 给 出 TrueType 字体 所 在 的 文件 夹 路 径 ， 如 
在 文件 开头 使 用 以 下 语句 : 

putenv ('GDFONTPATH=C:\WINDOWS\Fonts'); 

使 用 TrueType 字体 也 可 以 直接 使 用 imagettftext0 函 数 。 它 是 使 用 ttf 字体 的 imagestring() 
函数 。 其 语法 格式 如 下 : 

imagettftext (图 片 对 象 ， 字 体 大 小 ， 文 字 显示 角度 ， 起 始 x 坐标 ， 起 始 y 坐标 ， 文 字 颜 色 ， 字 体 

名 称 ， 文 字 信息 ) 

另外 ， 一 个 很 重要 的 问题 就 是 GD 库 中 的 imagettftextO 函 数 默认 是 无 法 支持 中 文字 符 并 添 
加 到 图 片上 去 的 。 这 是 因为 GD 库 的 imagettftext0 函 数 对 于 字符 的 编码 是 采用 的 UTF-8 的 编 
码 格式 ， 而 简体 中 文 的 默认 格式 为 GB2312。 

下 面 介 绍 这 样 的 一 个 例子 。 具 体操 作 步 又 如 下 。 

EGR 把 c\\WINDOWS\Fonts 下 的 字体 文件 simhei.ttf 复制 到 与 文件 12.4.php 相同 的 目 

录 下 。 

ES 在 网 站 目录 下 建立 12.4.php 文件 。 代 码 如 下 : 

<?php 

$ysize = 200; 

$xsize = 300; 

Stheimage = imagecreatetruecolor ($xsize, S$ysize); 


$color2 = imagecolorallocate ($theimage, 8,2,133); 
$color3 = imagecolorallocate ($theimage, 230,22,22); 
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imagefill($theimage, 0, 0, $color2); 

$fontname = 'simhei.ttf'; 

$zhtext = "这 是 一 个 把 中 文 用 黑体 显示 的 图 片 。"; 

$text = iconv("GB2312", "UTF-8", S$zhtext); 

imagettftext ($theimage,12,0,20,100,$color3, $fontname, $text); 
header ('content-type: image/png'); 

imagepng ($theimage); 

imagedestroy ($theimage); 

> 


运行 程序 ， 结 果 如 图 14-6 所 示 。 
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图 14-6 ”把 中 文 用 黑体 显示 的 图 片 
【案例 剖析 】 


(1) imagefill($theimage, 0, 0,$color2) 之 前 的 语句 是 创建 画布 、 填 充 颜 色 的 。 

(2) $fontname='simhei.ttf 语 句 确认 了 当前 目录 下 的 黑体 字 的 ttf 文件 ， 并 且 把 路 径 赋 值 给 
$fontname 变量 。 

(3) Szhtext 中 ， 中 文字 符 的 编码 为 GB2312。 为 了 转换 此 编码 为 UTF-8， 使 用 $text= 
iconv("GB2312","UTF-8"，S$zhtext) 语 句 把 Szhtext 中 的 中 文 编码 转换 为 UTF-8， 并 赋值 给 $text 
变量 。 

(4) imagettftext($theimage,12,0,20,100,$color3,$fontname,$text) 语 句 按照 imagettftextO 函 数 
的 格式 分 别 确认 了 参数 。S$theimage 为 目标 图 片 ，“12” 为 字符 的 大 小 ，“0” 为 显示 的 角 
度 ，“20,100” 为 字符 串 显示 的 初始 x、y 的 值 。$fontname 为 已 经 设 定 的 黑体 ，$text 为 已 经 
转换 为 UTF-8 格式 的 中 文字 符 串 。 





14.3 Jpgraph 库 的 基本 操作 


Jpgraph 是 一 个 功能 强大 且 十 分 流行 的 PHP 外 部 图 片 处 理 库 文件 。 它 是 建立 在 内 部 库 文件 
GD2 库 之 上 的 。 它 的 优点 是 建立 了 很 多 方便 操作 的 对 象 和 函数 ， 能 够 大 大 地 简化 使 用 GD 库 
对 图 片 进行 处 理 的 编程 过 程 。 
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14.3.1 Jpgraph 的 安装 


Jpgraph 的 安装 就 是 PHP 对 Jpgraph 类 库 的 调用 。 可 以 采用 多 种 形式 。 但 是 ， 首 先 都 需要 
到 Jpgraph 的 官方 网 站 下 载 类 库 文件 的 压缩 包 。 到 http://jpgraph.net/download/ 下 载 最 新 的 压缩 
包 ， 即 Jpgraph 4.0.2。 解 压 以 后 ， 如 果 是 Linux 系统 ， 可 以 把 它 放置 在 lib 目录 下 ， 并 且 使 用 
下 面 的 语句 重 命名 此 类 库 的 文件 夹 : 


ln -s jpgraph-4.x jpgraph 


如 果 是 Windows 系统 ， 在 本 机 WAMP 的 环境 下 ， 则 可 以 把 类 库 文件 夹 放 在 www 目录 
下 ,或 者 放置 在 项 目的 文件 夹 下 ， 如 图 14-7 所 示 。 
| 日 =|jpgraph - DO x 


EE ="  :*= ss © 


< v 不 目 。 windowsl0 (CC ，wamp » www » code ，ch14 » jpgraph v 局 当权 jpgraph” 帮 
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图 14-7 Jpgraph 库 的 文件 夹 


docs 2017/8/11 18:45 文件 实 
sre 2017/8/11 18:46 。 文件 夫 
] VERSION 2016/12116 11:12 ”文件 1 











其 中 各 个 文件 的 含义 如 下 。 

(1) docs 文件 夹 : 包含 jpgraph 的 开发 文档 。 

(2) src 文件 夹 : 图 表 生 成 所 依赖 的 代码 包 ， 其 子 目 录 Examples 里 有 许多 实例 。 

(3) src\Examples 文件 夹 : 里 面包 含 许多 实例 ， 使 用 它们 可 以 制作 各 种 各 样 的 图 表 。 

(4) docs\chunkhtml 文件 夹 ， 里面 有 许多 案例 及 附 有 图 表 。 

然后 在 程序 中 引用 的 时 候 ， 直 接 使 用 require_once0 命 令 ， 并 且 指 出 Jpgraph 类 库 相 对 于 
此 应 用 的 路 径 。 

在 本 机 环境 下 ， 把 jpgraph 文件 夹 放 置 在 C:\wamp\www\code\ch14 文件 夹 下 。 在 应 用 程序 
的 文件 中 加 载 此 库 ， 使 用 require_once (jpgraph/src/ijpgraph.php) 即 可 。 


14.3.2 ”Jpgraph 的 配置 


使 用 Jpgraph 类 前 ， 需 要 对 PHP 系统 的 一 些 限制 性 参数 进行 修改 。 具 体 修 改 以 下 3 个 方 
面 的 内 容 。 

(1) 需要 到 php.ini 中 修改 内 存 限制 ，memory_limit 至 少 为 32m， 本 机 环境 为 momery_ 
limit = 128m。 

(2) 最 大 执行 时 间 max_execution_time 要 增加 ， 即 max _execution time = 120。 

(3) 用 “;” 号 注释 掉 output buffering 选项 。 


14.4 “案例 实战 1 一 一 制作 圆 形 统计 


Jpgraph 库 安 装 设置 生效 以 后 ， 就 可 以 使 用 此 类 库 了 。 由 于 Jpgraph 有 很 多 示例 ， 所 以 读 
者 可 以 轻松 地 通过 示例 来 学 习 。 

下 面 就 通过 圆 形 统计 图 例子 的 介绍 ， 来 了 解 Ipgraph 类 的 使 用 方法 和 技巧 ， 具 体 步 骤 如 下 。 

ED 找到 安装 过 的 jpgraph 类 库 文件 夹 ， 在 其 下 的 src 文件 夹 下 找到 Examples 文件 

夹 。 找 到 balloonex1.php 文件 ， 将 其 复制 到 ch14 文件 夹 下 。 代 码 如 下 : 

<?php 

// content="text/plain; charset=utf-8" 

// $Id: balloonexl.php,v 1.5 2002/12/15 16:08:51 aditus Exp $ 

date default timezone set('Asia/Chongqing'); 

require once ('jpgraph/jpgraph.php'); 

require_once ('jpgraph//jpgraph_scatter.php'); 





// Some data 

$datax = array(l,2,3,4,5,6,7,8); 

$datay = array(12,23,95,18,65,28,86,44); 

// Callback for markers 

// Must return array (width,color,fill color) 

// If any of the returned values are "" then the 

// default value for that parameter will be used. 

function FCallback($aVal) { 
// This callback will adjust the fill color and size of 
// the datapoint according to the data value according to 
if( $aVal < 30 ) $c = "blue"; 
elseif( $aVal < 70 ) $c = "green"; 
else $c="red"; 
return array (floor ($aVal/3),"",$c); 
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} 


// Setup a basic graph 

$graph = new Graph (400,300,'auto'); 

$graph->Setscale ("linlin"); 

$graph->img->SetMargin (40,100,40,40); 
$graph->SetShadow (); 

$graph->title->Set ("Example of ballon scatter plot"); 
// Use a lot of grace to get large scales 
S$graph->yaxis->scale->SetGrace (50,10); 


// Make sure X-axis as at the bottom of the graph 
$graph->xaxis->SetPos('min'); 


// Create the scatter plot 
$spl = new ScatterPlot ($datay, $datax); 
$spl->mark->SetType (MARK_FILLEDCIRCLE); 


// Uncomment the following two lines to display the values 
$spl->value->Show (); 
$spl->value->SetFont (FF_FONT1, FS_BOLD); 
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// Specify the callback 
$spl->mark->SetCallback ("FCallback"); 


// Setup the legend for plot 
$spl->SetLegend('Year 2002'); 


// Add the scatter plot to the graph 
$graph->Add ($sp1); 


// ... and send to browser 
$graph->stroke (); 


> 


ER 修改 require_once(jpgraph/ipgraphphp) 为 require_ once(jpgraph/src/jpgraph php)。 修 改 
require_once(jpgraph/ipgraph_scatter.php) 为 require_once(jpgraph/srcipgraph_scatterphp)。 以 载 
入 本 机 Jpgraph 类 库 。 

运行 balloonex1.php， 结 果 如 图 14-8 所 示 。 
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14-8 balloonex1.php 页 面 的 效果 
【案例 剖析 】 


(1) require_once(jpgraph/src/jpgraph.php) 语 句 和 require_once('jpgraph/src/jpgraph scatter. 
php) 语 名 加载 了 Jpgraph 基本 类 库 jpgraph.php 和 圆 形 图 类 库 jpgraph_bar.php。 

(2) $datax 和 $datay 定义 了 两 组 要 表现 的 数据 。 

(3) function FCallback($aVal){} 函 数 定义 了 不 同 数值 范围 内 的 图 形 的 颜色 。 

(4) $graph = new Graph(400,300,'auton) 语 句 生 成 图 形 。$graph->SetScale("linlin") 生 成 刻 
度 。S$graph->img->SetMargin(40,100,40,40) 设 置 图 形 边框 。$graph->SetShadow() 设 置 阴影 。 
$egraph->title->Set("Example of ballon scatter plot") 设 置 标题 。$graph->xaxis->SetPos('min'); 设 置 
x 轴 的 位 置 为 初始 值 。 

(5) $spl = new ScatterPlot($datay.$datax) 生 成 数据 表示 图 。$sp1->mark->SetType(MARK 
FILLEDCIRCLE) 设 置 数据 表示 图 的 类 型 。$sp1->value->Show0O 展 示 数 据 表 示 图 。$sp1-> value- 


>SetFont(FF_FONTI1,FS_BOLD) 设 定 展示 图 的 字体 。$sp1->SetLegend('Year 2002) 设 置 标题 。 
(6) S$graph->Add($sp1) 添 加 数据 展示 图 到 整体 图 形 中 。 
(7) $graph->StrokeO) 语 句 表 示 把 此 图 传递 到 浏览 器 显示 。 


14.5 “案例 实战 2 一 一 制作 3D 饼 形 统计 图 


下 面 就 通过 3D 饼 形 图 例 程 的 介绍 ， 来 了 解 pgraph 类 的 使 用 方法 和 技巧 。 
EC 找到 安装 过 的 jpgraph 类 库 文件 夹 ， 在 其 下 的 sre 文件 夹 下 找到 Examples 文件 
夹 。 找 到 pie3dex3.php 文件 ， 将 其 复制 到 ch14 文件 夹 下 。 打 开 查 看 ， 代 码 如 下 : 


<?php 

require_once('jpgraph/ jpgraph.php'); 
require_once('jpgraph/jpgraph pie.php'); 
require_once('jpgraph/jpgaph pie3d.php'); 
$data = array (20,27,45,75,90); 

$graph = new PieGraph (450,200); 
$graph->SetShadow() 7 

$graph->title->Set ("Example 1 3D Pie plot"); 
S$graph->title->SetFont (FF_VERDANA, FS_BOLD,18); 
$graph->title->SetColor ("darkblue"); 
$graph->legend->Pos (0.5,0.8); 

$pl = new PiePlot3d($data); 

$pl->SetTheme ("sand"); 

$pl->SetCenter (0.4); 

$pl->SetAngle (30); 

$pl->value->SetFont (FF_ARIAL, FS_NORMAL, 12); 
$pl->SetLegends (array ("Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", 
"Aug", "Sep", "Oct")); 

$graph->Add ($p1); 

$graph->Stroke () 7 

区 人 


修改 require_once(jpgraph/ipgraph.php) 为 require_once(jpgraph/src/jpgraphphp)。 修 改 
require_once(jpgraph/ipgraph pie php) 为 require_once(jpgraph/src/ipgraph piephp) 。 修 改 
require_once('jpgraph/jpgraph_pie3d.php') 为 require_once(jpgraph/src/ipgraph pie3dphp) 。 
目的 是 载 入 本 机 Jpgraph 类 库 。 

运行 pie3dex3.php， 结 果 如 图 14-9 所 示 。 
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图 14-9 pie3dex3.php 页 面 的 效果 
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【案例 剖析 】 


(1) require_ once(jpgraph/src/ipgraphphp) 语 句 、 require_ once(jpgraph/ipgraph pie.php') 和 
require_once(jpgraph/jpgraph_pie3d.php) 语 句 加 载 了 Jpgraph 基本 类 库 jpgraph.php、 饼 形 图 类 
库 jpgraph_pie.php 和 3d 饼 形 图 类 库 jpgraph_pie3d.php。 

(2) $data 定义 了 要 表现 的 数据 。 

(3) $graph = new PieGraph(450,200) 生 成 图 形 。$graph->SetShadowO 设 定 阴 影 。 

(4) $graph->title->Set("Example 1 3D Pie plot") 设 定 标 题 。 S$graph->title->SetFont(FF_ 
VERDANA,FS_BOLD,18) 设 定 字体 和 字体 大 小 。$graph->title->SetColor("darkblue") 设 定 颜 色 。 
Sgraph->legend->Pos(0.5,0.8) 设 定 图 例 在 整个 图 形 中 的 位 置 。 

(5) $pl = new PiePlot3d($data) 生 成 饼 形 图 。$p1->SetTheme(sand) 设 置 饼 形 图 模板 。$p1- 
”>SetCenter(0.4) 设 置 饼 形 图 的 中 心 。$pl->SetAngle(30) 设 置 饼 形 图 角度 。S$pl->value-> 
SetFont(FF_ARIAL,FS NORMAL,12) 设 置 字体 。S$p1->SetLegends(array("Jan",.…,"Oct")) 设 置 





例文 字 信息 。 
(6) $graph->Add($p1) 向 整个 图 形 添加 饼 形 图 。$graph->StrokeO0 把 此 图 传递 到 浏览 器 进行 
显示 。 


14.6 疑难 解 惑 


疑问 1: 在 制作 圆 形 统计 图 时 ， 报 出 很 多 错误 和 警告 ， 如 图 14-10 所 示 ， 如 何 解决 ? 
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图 14-10 ”报错 页 面 
: 在 圆 形 统计 图 文件 源 代码 第 一 行 ， 内 容 如 下 : 


咀 
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// content="text/plain; charset=utf-8" 
这 里 规定 编码 方式 必须 为 tf8， 所 以 需要 在 php.ini 文件 中 设置 默认 编码 方式 ( 见 图 14-11): 
default_charset="utf-8" 


然后 重启 服务 器 即 可 生效 。 


目 phpini - 书本 - oO x 
文生 ”篇 基 日 、 闪 式 O) 下 看 M 大 
auto_append_file = 














: By default, PHP will output a media type using the Content-Type header. To 
; dsable this, simply set it to be enpty. 





; PHF’ 3 built-in default ia set to texrt/htnl. 
‘http: /pop. ault- 
cet 


1 PHF’s default character sat is set to UTF-8. 


+t is st 
a "+ 





ctor encoding is ser to ompty. 
t_charset is used. 
ut-sncoding 








Input_encoding 





,PHP outpat cha 
ST default_。 


5 output_bu 
http://pap. net/output-encoding 
encoding = 





图 14-11 设置 默认 编码 方法 
疑问 2: 不 同 格式 的 图 片 使 用 上 有 何 区 别 ? 


答 : JPEG 格式 是 一 个 标准 。JPEG 经 常用 来 储存 照片 和 拥有 很 多 颜色 的 图 片 ， 它 不 强调 
压缩 ， 强 调 的 是 对 图 片 信息 的 保存 。 如 果 使 用 图 形 编辑 软件 缩小 了 PEG 格式 的 图 片 ， 那 么 它 
原本 包含 的 一 部 分 数据 就 会 丢失 。 并 且 这 种 数据 的 丢失 通过 肉眼 是 可 以 察觉 到 的 。 这 种 格式 
不 适合 包含 简单 图 形 颜 色 或 文字 的 图 片 。 

PNG 格式 是 指 portable network graphics， 这 种 图 片 格式 是 发 明 出 来 以 取代 GIF 格式 的 。 
同样 的 图 片 使 用 PNG 格式 的 大 小 要 小 于 使 用 GIF 格式 的 大 小 。 这 种 格式 是 一 种 低 损失 压缩 的 
网 络 文件 格式 。 这 种 格式 的 图 片 适合 于 包含 文字 、 直 线 或 者 色 块 的 信息 。PNG 支持 透明 、 件 
马 校 正 等 。 但 是 PNG 不 像 GIF 一 样 支持 动画 功能 。 并 且 正 6 不 支持 PNG 的 透明 功能 。 低 损 
压缩 意味 着 压缩 比 不 高 ， 所 以 它 不 适合 用 于 照片 这 一 类 的 图 片 ， 否 则 文件 将 太 大 。 

GIF 是 指 graphics interchange format， 它 也 是 一 种 低 损 压 缩 的 格式 ， 适 合用 于 包含 文字 、 
直线 或 者 色 块 信息 的 图 片 。 它 使 用 的 是 24 位 RGB 色彩 中 的 256 色 。 由 于 色彩 有 限 ， 所 以 也 
不 适合 用 于 照片 一 类 的 大 图 片 。 对 于 其 适合 的 图 片 ， 它 具有 不 丧失 图 片 质量 却 能 大 幅 压 缩 的 
图 片 大 小 的 优势 。 另 外 ， 它 支持 动画 。 


疑问 3: 如 何 选择 自己 想 要 的 RGB 颜色 呢 ? 


答 : 可 以 使 用 Photoshop 里 面 的 颜色 选取 工具 。 如 果 使 用 的 是 Linux 系统 ， 可 以 使 用 开源 
的 工具 GIMP 中 的 颜色 选取 工具 。 
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第 15 章 
保持 HTTP 连接 状态 
一 一 Cookie 和 
会 话 管理 


HTTP Web 协议 是 无 状态 协议 ， 对 于 事务 处 理 没有 记忆 能 力 。 缺 少 状态 意味 着 
如 果 后 续 处 理 需 要 前 面 的 信息 ， 则 它 必须 重 传 ， 这 样 可 能 导致 每 次 连接 传送 的 数据 
量 增 大 。 客 户 端 与 服务 器 进行 动态 交互 的 Web 应 用 程序 出 现 之 后 ，HTTP 无 状态 的 
特性 严重 阻碍 了 这 些 应 用 程序 的 实现 。 毕 竟 交 互 是 需要 承前启后 的 ， 简 单 的 购物 车 
程序 也 要 知道 用 户 到 底 在 先前 选择 了 什么 商品 。 于 是 ， 两 种 用 于 保持 HITP 连接 状 
态 的 技术 就 应 运 而 生 了 ， 一 个 是 Cookie， 而 另 一 个 则 是 Session。 其 中 Cookie 将 数 
据 存 储 在 客户 端 ， 并 显示 永久 的 数据 存储 。Session 将 数据 存储 在 服务 器 端 ， 保 证 
数据 在 程序 的 单 次 访问 中 持续 有 效 。 本 章 主 要 讲述 Cookie 和 Session 的 使 用 方法 和 
应 用 技巧 。 
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15.1 ”Cookie 的 基本 操作 
下 面 介绍 Cookie 的 含义 和 基本 用 法 。 


15.1.1 什么 是 cookie 


Cookie 常用 于 识别 用 户 。Cookie 是 服务 器 留 在 用 户 计 算 机 中 的 小 文件 。 

Cookie 的 工作 原理 是 : 当 一 个 客户 端 浏览 器 连接 到 一 个 URL 时 ， 它 会 首先 扫描 本 地 储存 
的 Cookie， 如 果 发 现 其 中 有 与 此 URL 相关 联 的 Cookie， 将 会 把 它 返 回 给 服务 器 端 。 

Cookie 通常 应 用 于 以 下 几 个 方面 。 

(1) 在 页 面 之 间 传递 变量 。 因 为 浏览 器 不 会 保存 当前 页 面 上 的 任何 变量 信息 ， 如 果 页 面 被 
关闭 ， 则 页 面 上 的 所 有 变量 信息 也 会 消失 。 而 通过 Cookie， 可 以 把 变量 值 在 Cookie 中 保存 下 
来 ， 然 后 另外 的 页 面 就 可 以 重新 读 取 这 个 值 。 

(2) 记录 访客 的 一 些 信息 。 利 用 Cookie， 可 以 记录 客户 曾经 输入 的 信息 ， 或 者 记录 访问 
网 页 的 次 数 。 

(3) 通过 把 所 查看 的 页 面 存 放 在 Cookie 临时 文件 夹 中 ， 可 以 提高 以 后 的 浏览 速度 。 

用 户 可 以 通过 header 以 如 下 格式 在 客户 端 生成 Cookie: 


Set-cookie:NAME=VALUE; [expires=DATE; ] [path=PATH; ] [domain=DOMAIN_NAME; ] [secure] 


NAME 为 Cookie 名 称 ; VALUE 为 Cookie 的 值 ; expires=DATE 为 到 期 日 ; 
path=PATH 、domain=DOMAIN NAME 为 与 某 个 地 址 相对 应 的 路 径 和 域名 ; secure 表示 
Cookie 不 能 通过 单一 的 HTTP 连接 传递 。 


15.1.2 创建 Cookie 


通过 PHP， 用 户 能 够 创建 Cookie。 创 建 Cookie 使 用 PHP 的 setcookie0 函 数 ， 它 的 语法 格 
式 如 下 : 
setcookie (名 称 , Cookie 值 , 到 期 日 , 路径, 域名, secure) 


其 中 的 参数 与 Set-cookie 中 的 参数 意义 相同 。 


Sn setcookie() 阴 数 必须 位 于 <html> 标 签 之 前 。 


在 下 面 的 例子 中 ， 将 创建 名 为 user 的 Cookie， 把 它 赋值 为 “Cookie 保存 的 值 ”， 并 且 规 
定 了 此 Cookie 在 1 小 时 后 过 期 。 
【 例 1S.1】 使 用 setcookieO) 函 数 (示例 文件 ch15\15.1.php)。 


<?php 

setcookie ("user"，"Cookie 保存 的 值 "，time ()+3600); 
?> 

<html> 


<body> 

</body> 

</html> 

运行 上 述 程序 ， 会 在 cookies 文件 夹 下 自动 生成 一 个 Cookie 文件 ， 有 效 期 为 1 个 小 时 ， 
在 Cookie 失效 后 ，Cookies 文件 将 自动 被 删除 。 


Sm 如 果 用 户 没 有 设置 Cookie 的 到 期 时 间 ， 则 默认 立即 到 期 ， 即 在 关闭 浏览 器 时 会 
SS 自动 删除 Cookie 数据 。 


15.1.3 读 取 Cookie 


那么 ， 如 何 取 回 Cookie 的 值 呢 ? 在 PHP 中 ， 使 用 $_COOKIE 变量 取 回 Cookie 的 值 。 下 
面 通过 示例 讲解 如 何 取 回 上 面 创建 的 名 为 user 的 Cookie 的 值 ， 并 把 它 显 示 在 页 面 上 。 

【 例 15.2】 读 取 cookie( 示 例文 件 ch15\15.2.php)。 

<?php 

// 输出 一 个 cookie 

echo $_COOKIE["user"]; 

// 显示 所 有 的 cookie 


print_r($_ COOKIE); 
人 


程序 运行 结果 如 图 15-1 所 示 。 
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15-1 读 取 Cookie 


用 户 可 以 通过 isset() 函 数 来 确认 是 否 已 设置 了 Cookie。 下 面 通过 示例 来 讲解 。 
【 例 15.3】 使 用 isset0 函 数 来 确认 是 否 已 设置 了 cookie( 示 例文 件 ch15\15.3.php)。 


<html> 

<body> 

<?php 

if (isset($ COOKIE["user"])) // 假 如 cookie 文件 存在 
echo "Welcome " . $ COOKIE["user"] . "!<br />"; 

else // 如 果 cookie 文件 不 存在 
echo "Welcome guest!<br />"; 

?> 

</body> 

</html> 


程序 运行 结果 如 图 15-2 所 示 。 
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15-2 ”通过 isset() 函 数 来 确认 是 否 已 设置 了 Cookie 
15.1.4 删除 Cookie 


常见 的 删除 Cookie 的 方法 有 两 种 ， 即 在 浏览 器 中 手动 删除 和 使 用 函数 删除 。 
1. 在 浏览 器 中 手动 删除 


由 于 Cookie 自动 生成 的 文本 会 存在 于 下 浏览 器 的 cookies 临时 文件 夹 中 ， 在 浏览 器 中 删 
除 Cookie 文件 是 比较 快捷 的 方法 。 具 体操 作 步 又 如 下 。 
ES 在 浏览 器 的 菜单 栏 中 选择 “工具 ”一 “Internet 选项 ”命令 ， 如 图 15-3 所 示 。 








层 htpylocahosycode/ch P ~ ©|| Blocalhost x 介 交 次 图 
文 #( 六 (5) 下 在) «mA) EE wa 


ie 保存 的 佳丽 尖 党 历史 局 录 D)- cel:Shit+ Del 
elcome Cookie 保存 的 舍 。 8 | 


任 间 仪表 板 Cul+Shift+U 











| 允许 你 更 妆 设 置 。 
图 15-3 选择 “Internet 选项 ”命令 
ERP 弹出 “Intermet 选项 ”对 话 框 ， 然 后 在 “常规 ”选项 卡 中 单 击 “ 删 除 ” 按 钮 ， 如 
图 15-4 所 示 。 
ES 弹出 “删除 浏览 历史 记录 ”对 话 框 ， 勾 选 相应 的 复 选 框 ， 单 击 “ 删 除 ”按钮 即 
可 ， 如 图 15-5 所 示 。 返 回 到 “Intemet 选项 ”对 话 框 ， 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 
删除 Cookie 的 操作 。 
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图 15-4 “Internet 选项” 对话 框 图 15-5 “删除 浏览 历史 记录 ”对 话 框 
2. 使 用 函数 删除 


删除 Cookie 仍然 使 用 setcookie0) 函 数 。 当 删除 cookie 时 ， 将 第 二 个 参数 设置 为 空 ， 第 三 
个 参数 的 过 期 时 间 设 置 为 小 于 系统 的 当前 时 间 即 可 。 
【 例 15.4】 使 用 setcookieO 函 数 删除 cookie( 示 例文 件 ch15\15.4.php)。 


<?php 

/将 Cookie 的 过 期 时 间 设 置 为 比 当前 时 间 减 少 10 秒 
setcookie("user", "", 七 ime()-10) 

> 


在 上 述 代 码 中 ，time0 函 数 返回 的 是 当前 的 系统 时 间 ， 把 过 期 时 间 减 少 10 秒 ， 这 样 过 期 
时 间 就 会 变 成 过 去 的 时 间 ， 从 而 删除 Cookie。 如 果 将 过 期 时 间 设 置 为 0， 则 也 可 以 直接 删除 
Cookie。 
15.2 认识 Session 
下 面 介绍 Session 的 基本 概念 和 使 用 方法 。 
15.2.1 什么 是 Session 


由 于 HTTP 是 无 状态 协议 ， 也 就 是 说 ，HTTP 的 工作 过 程 是 请 求 与 回应 的 简单 过 程 ， 所 以 
HTTP 没有 一 个 内 置 的 方法 来 储存 在 这 个 过 程 中 各 方 的 状态 。 例 如 ， 当 同一 个 用 户 向 服务 器 发 
出 两 个 不 同 的 请 求 时 ， 虽 然 服务 器 端 都 会 给 予 相应 的 回应 ， 但 是 它 并 没有 办 法 知道 这 两 个 动 
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作 是 由 同一 个 用 户 发 出 的 。 
由 此 ， 会 话 (Session) 管 理应 运 而 生 。 通 过 使 用 一 个 会 话 ， 程 序 可 以 跟踪 用 户 的 身份 和 行 
为 ， 并 且 根 据 这 些 状态 数据 ， 给 用 户 以 相应 的 回应 。 


15.2.2 ”Session 的 基本 功能 


在 PHP 中 ， 每 一 个 Session 都 有 一 个 ID。 这 个 Session ID 是 一 个 由 PHP 随机 生成 的 加 密 
数字 。 这 个 Session ID 通过 Cookie 储存 在 客户 端 浏览 器 中 ， 或 者 直接 通过 URL 传递 至 客户 
端 。 如 果 在 某 个 URL 后 面 看 到 一 长 串 加 密 的 数字 ， 这 很 有 可 能 就 是 Session ID 了 。 

Session ID 就 像 是 一 把 钥匙 ， 用 来 注册 到 Session 变量 中 。 而 这 些 Session 变量 是 储存 在 服 
务 器 端的 。Session ID 是 客户 端 唯一 存在 的 会 话 数据 。 

使 用 Session ID 打开 服务 器 端 相对 应 的 Session 变量 ， 跟 用 户 相关 的 会 话 数据 便 一 目 了 
然 。 在 默认 情况 下 ， 在 服务 器 端的 Session 变量 数据 是 以 文件 的 形式 加 以 储存 的 ， 但 是 会 话 变 
量 数据 也 经 常 通过 数据 库 进行 保存 。 


15.2.3 ”Cookie 与 Session 


在 浏览 器 中 ， 有 些 用 户 出 于 安全 性 的 考虑 ， 关 闭 了 其 浏览 器 的 Cookie 功能 ， 导 致 Cookie 
不 能 正常 工作 。 

使 用 Session 可 以 不 需要 手动 设置 Cookie，PHP Session 可 以 自动 处 理 。 可 以 使 用 会 话 管 
理 及 PHP 中 的 session_get_cookie_params0 〇 函数 来 访问 Cookie 的 内 容 。 这 个 函数 将 返回 一 个 
数组 ， 包 括 Cookie 的 生存 周期 、 路 径 、 域 名 、secure 等 。 它 的 语法 格式 如 下 : 


session get_cookie params (生存 周期 ， 路 径 ， 域名 :secure) 


15.2.4 储存 Session ID 在 Cookie 或 URL 中 


PHP 在 默认 情况 下 会 使 用 Cookie 来 储存 Session ID。 但 是 如 果 客 户 端 浏览 器 不 能 正常 工 
作 ， 就 需要 用 URL 方式 传递 Session ID 了 。 把 php.ini 中 的 session.use_trans_sid 设置 为 启用 状 
态 ， 就 可 以 自动 通过 URL 来 传递 Session ID 。 

不 过 ， 通 过 URL 传递 Session ID 会 产生 一 些 安全 问题 。 如 果 这 个 连接 被 其 他 用 户 拷贝 并 
使 用 ， 有 可 能 造成 用 户 判断 的 错误 。 其 他 用 户 可 能 使 用 Session ID 访问 目标 用 户 的 数据 。 

或 者 可 以 通过 程序 把 Session ID 储存 到 常量 SID 中 ， 然 后 通过 一 个 连接 传递 。 


15.3 会 话 管理 


一 个 完整 的 会 话 包括 创建 会 话 、 注 册 会 话 、 使 用 会 话 和 删除 会 话 。 下 面 介绍 有 关 会 话 管 
理 的 基本 操作 。 


15.3.1 创建 会 话 


常见 的 创建 会 话 的 方法 有 3 种 ， 包 括 PHP 自动 创建 、 使 用 session_startO 函 数 创建 和 使 用 
session_register() 函 数 创建 。 

1. PHP 自动 创建 

用 户 可 以 在 php.ini 中 设 定 session.auto_start 为 启用 。 但 是 ， 使 用 这 种 方法 的 同时 ， 不 能 
把 Session 变量 对 象 化 。 应 定义 此 对 象 的 类 必须 在 创建 会 话 之 前 加 载 ， 然 后 新 创建 的 会 话 才能 
加 载 此 对 象 。 

2. 使 用 session start( 函 数 

这 个 函数 首先 会 检查 当前 是 否 已 经 存在 一 个 会 话 ， 如 果 不 存在 ， 它 将 创建 一 个 全 新 的 会 
话 ， 并 且 这 个 会 话 可 以 访问 超 全 局 变量 $_ SESSION 数组 。 如 果 已 经 有 一 个 存在 的 会 话 ， 函 数 
会 直接 使 用 这 个 会 话 ， 加 载 已 经 注册 过 的 会 话 变量 ， 然 后 使 用 。 

session_startO) 函 数 的 语法 格式 如 下 : 


bool session_start (void) 7 


Session_startO 函 数 必须 位 于 <html> 标 签 之 前 。 


削 嘱 恋 沙 过 8Mo0o0 一 一 济 准 河 了 及 dLLH 注视 出 9 小 便 
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【 例 15.5】 使 用 session_start0 函 数 ( 示 例文 件 ch15\15.5.php)。 


<?php session start(); ?> 
<html> 
<body> 
</body> 
</html> 


上 述 代码 会 向 服务 器 注册 用 户 的 会 话 ， 以 便 可 以 开始 保存 用 户 信息 ， 同 时 会 为 用 户 会 话 
分 配 一 个 WD。 


3. 使 用 session registerO) 函 数 

在 使 用 session_register0 函 数 之 前 ， 需 要 在 php.ini 文件 中 将 register_globals 设置 为 on， 然 
后 需要 重启 服务 器 。session_register() 函 数 通 过 为 会 话 登记 一 个 变量 来 隐 含 地 启动 会 话 。 
15.3.2 ”注册 会 话 变量 

会 话 变量 被 启动 后 ， 全 部 保存 在 数组 $ SESSION 中 。 用 户 可 以 通过 对 $_SESSION 数组 赋 


值 来 注册 会 话 变量 。 
例如 ， 启 动 会 话 ， 创 建 一 个 Session 变量 ， 并 赋予 “xiaoli” 的 值 ， 代 码 如 下 : 
<?php 
session start(); // 启 动 Session 
$_SESSION['name']='xiaoli'; // 声 明 一 个 名 为 name 的 变量 ， 并 赋值 “xiaoli” 
E22 
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这 个 会 话 变量 值 会 在 此 会 话 结束 或 被 注销 后 失效 ， 或 者 还 会 根据 php.ini 中 的 session.gc_ 
maxlifetime( 当 前 系统 设置 为 1440 秒 ， 也 就 是 24 小 时 ) 会 话 最 大 生命 周期 数 过 期 而 失效 。 


15.3.3 ”使 用 会 话 变量 


使 用 会 话 变量 ， 首 先 要 判断 会 话 变量 是 否 存在 一 个 会 话 ID 。 如 果 不 存在 ， 则 需要 创建 一 
个 ， 并 且 能 够 通过 $_SESSION 变量 进行 访问 。 如 果 已 经 存在 ， 则 将 这 个 已 经 注册 的 会 话 变 量 
载 入 ， 以 供用 户 使 用 。 

在 访问 $_SESSION 数组 时 ， 先 要 使 用 isset0 或 empty0 来 确定 $_ SESSION 中 会 话 变量 是 否 
为 宝 z 





例如 : 

<?php 

if(!empty($_SESSION['session name'])) // 判 断 会 话 变量 是 否 为 空 
$ssvalue = $_SESSION['session name']; // 声 明 一 个 变量 并 赋值 


?> 


下 面 通过 例子 来 讲解 存储 和 取 回 $_SESSION 变量 的 方法 。 
【 例 15.6】 存 储 和 取 回 $_SESSION 变量 (示例 文件 ch15\15.6.php)。 











<?php 

session_start (); 

// 存 储 会 话 变量 的 值 

$_SESSION['views'] = 1; 

区 六 

<html> 

<body> 

<?php 

// 读 取 会 话 变量 的 值 

echo "浏览 量 =". $_SESSION['views']; 

EE 

</body> 

</html> 

程序 运行 结果 如 图 15-6 所 示 。 = 

@ 你 hapyilocalhosycode P - 0 | 确 localhos 

15.3.4 “注销 和 销毁 会 话 变量 Ee ae ee 

注销 会 话 变量 使 用 unset0 函数 就 可 以 ， 如 
unset($_SESSION[mame'"])( 不 再 需要 使 用 PHP 4 中 的 
session_unregister() 或 session unsetO 了 )。 15-6 ”存储 和 取 回 $_ SESSION 变量 

unset0 函 数 用 于 释放 指定 的 Session 变量 。 代 码 如 下 : 

<?php 


unset ($_SESSION['views']); 
2 


如 果 要 注销 所 有 会 话 变 量 ， 只 需要 向 $_SESSION 赋值 一 个 空 数组 就 可 以 了 ， 例 如 
$_SESSION = amay0。 注 销 完成 后 ， 使 用 session_destroy0 销 毁 会 话 即 可 ， 其 实 就 是 清除 相应 
的 Session ID 。 代 码 如 下 : 
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<?php 
session destroy(); 
?> 


15.4 ”案例 实战 一 一 综合 应 用 会 话 管理 


下 面 通过 一 个 综合 案例 讲述 会 话 的 综合 应 用 。 
在 网 站 根 目录 下 建立 一 个 session 文件 夹 。 
在 session 文件 夹 下 建立 opensession.php， 输 入 以 下 代码 并 保存 : 


<?php 

session_ start (); 

$_SESSION['name'] = "王小明 "; 

echo "会 话 变量 为 :".$_SESSION['name']; 
EE 

<a href='usesession.php'> 下 一 页 </a> 


在 session 文件 夹 下 建立 usesession.php 文件 ， 输 入 以 下 代码 并 保存 : 


<?php 

session_start (); 

echo "会 话 变量 为 :".$_SESSION ['name'] ."<br />"; 
echo $_SESSION['name'].", 你 好 。" 

人 > 

<a href='closesession.php'> 下 一 页 </a> 


在 session 文件 夹 下 建立 closesession.php 文件 ， 输 入 以 下 代码 并 保存 : 


<?php 

session start (); 

unset ($_SESSION['name']); 

if (isset($_SESSION['name']))1{ 

echo "会 话 变量 为 :" .$_SESSION['name']; 
}else{ 

echo "会 话 变量 已 注销 。" 

session destroy(); 

> 


运行 opensession.php 文件 ， 结 果 如 图 15-7 所 示 。 
ED 单 击 页 面 中 的 “下 一 页 ”链接 ， 结 果 如 图 15-8 所 示 。 


削 趴 束 钞 普 anloo0 一 一 斌 尖 六 订 dLLH 六 关山 9 小 


























一 口 X 口 x 
人 @ httpi//localhost/code PD - 上 | @ localhost @ [Es httpi//localho.. P ~ © | 七 localhost 
文件 (月 ”篇 缉 (E) 可 看 (V) 收藏 夫 (A) 工具 (T) 帮助 (H) 文件 (有) 往生 (6。 查看 (V) ”收藏 次 (A) ”工具 (T) 帮助 (H) 
会 话 变量 为 :王小明 下 一 页 会 话 变量 为 :王小明 
王小明 , 你 好 。 斑 一 页 
夸 125% ~ 125% > 
图 15-7 ”程序 初始 结果 图 15-8 单 击 链接 后 的 结果 


EGR 继续 单 击 页 面 中 的 “下 一 页 ”链接 ， 结 果 如 图 15-9 所 示 。 
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be 口 x 
[BI@ hpy/ocalho.. 只 - © | @localhost 
文件 由 志和 (E) 可 看 V) 收 共 A) 工具 才 色 () 
会 话 变量 已 注销 . 

















坑 125% > 


15-9 ”会 话 变量 已 注销 


15.5 疑难 解 惑 


疑问 1: 如 果 浏 览 器 不 支持 Cookie， 该 怎么 办 ? 


答 : 如 果 应 用 程序 涉及 不 支持 Cookie 的 浏览 器 ， 不 得 不 采取 其 他 方法 在 应 用 程序 中 从 一 
个 页 面向 另 一 个 页 面 传递 信息 。 其 中 一 种 方式 就 是 从 表单 传递 数据 。 
下 面 的 表单 在 用 户 单 击 “ 提 交 ” 按 钮 时 向 welcome.php 提交 用 户 输入 : 


<html> 

<body> 

<form action="welcome.php" method="post"> 
Name: <input type="text" name="name" /> 
Age: <input type="text" name="age" /> 
<input type="submit" /> 

</form> 

</body> 

</html> 


要 取 回 welcome.php 中 的 值 ， 可 以 使 用 如 下 代码 : 


<html> 

<body> 

Welcome <?php echo $_POST["name"]; ?>.<br /> 
You are <?php echo $_POST["age"]; ?> years old. 
</body> 

</html> 











疑问 2: Cookie 的 生命 周期 是 多 久 ? 


答 : 如 果 Cookie 不 设 定 失效 时 间 ， 则 表示 它 的 生命 周期 为 未 关闭 浏览 器 前 的 时 间 段 ， 一 
旦 浏览 器 关闭 ，Cookie 会 自动 消失 。 

如 果 设 定 了 过 期 时 间 ， 那 么 浏览 器 会 把 Cookie 保存 到 硬盘 中 ， 在 超过 有 效 期 前 ， 用 户 打 
开 正 浏览 器 时 会 依然 有 效 。 

由 于 浏览 器 最 多 存储 300 个 Cookie 文件 ， 每 个 Cookie 文件 最 大 支持 4KB， 所 以 一 旦 超 
过 容量 的 限制 ， 浏 览 器 就 会 自动 随机 地 删除 Cookies。 


使 230 


第 3 篇 


操作 MySQL 数据 库 


第 16 章 


第 17 章 
第 18 章 
第 19 章 


管理 MySQL 的 利器 一 一 phpMyAdmin 操 
作 MySQL 数据 库 


数据 库 编 程 一 一 MySQL 数据 库 与 SQL 查询 
最 经 典 的 方法 一 一 使 用 MySQLi 操作 MySQL 
最 兼容 的 方法 一 一 使 用 PDO 操作 MySQL 





数据 库 


第 16 章 
管理 MySQL 的 利器 
一 -phpMyAdmin 

操作 MySQL 

数据 库 


要 使 一 个 网 站 达到 互动 效果 ， 不 是 让 网 页 充满 了 动画 和 音乐 ， 而 是 当 浏 览 者 对 
网 页 提出 要 求 时 能 出 现 响 应 的 结果 。 这 样 的 效果 大 多 需要 搭配 数据 库 的 使 用 ， 让 网 
页 读 出 保存 在 数据 库 中 的 数据 ， 显 示 在 网 页 上 。 因 为 每 个 浏览 者 对 于 某 个 相同 的 网 
页 所 提出 的 要 求 不 同 ， 显 示 出 的 结果 也 不 同 ， 这 才 是 真正 的 互动 网 站 。 由 于 
WampServer 集成 环境 已 经 安装 好 了 MySQL 数据 库 ， 通 过 phpMyAdmin 即 可 管理 
MySQL 数据 库 ， 更 重要 的 是 ， 操 作 非 常 简单 。 
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16.1 什么 是 MySQL 


MySQL 是 一 个 小 型 关系 数据 库 管理 系统 ， 与 其 他 大 型 数据 库 管理 系统 (如 Oracle、DB2、 
SQL Server 等 ) 相 比 ，MySQL 规模 小 、 功 能 有 限 ， 但 是 它 体积 小 、 速 度 快 、 成 本 低 ， 且 提供 
的 功能 对 稍微 复杂 的 应 用 来 说 已 经 够 用 。 这 些 特性 使 得 MySQL 成 为 世界 上 最 受 欢 迎 的 开放 
源 代 码 数据 库 。 


16.1.1 客户 -服务 器 软件 


主 从 式 结构 (Client/Server Model) 或 客户 -服务 器 (ClientServenD 结 构 ， 简 称 C/S 结构 ， 是 一 
种 网 络 架构 。 通 常 在 该 网 络 架 构 下 ， 软 件 分 为 客户 (Client) 和 服务 器 (Server) 两 个 部 分 。 

服务 器 是 整个 应 用 系统 资源 的 存储 和 管理 中 心 ， 多 个 客户 端 则 各 自 处 理 相 应 的 功能 ， 共 
同 实现 完整 的 应 用 。 在 客户 /服务 器 结构 中 ， 客 户 端 用 户 的 请 求 被 传送 到 数据 库 服 务 器 ， 数 据 
库 服务 器 进行 处 理 后 ， 将 结果 返回 给 用 户 ， 从 而 减少 了 网 络 数据 传输 量 。 

用 户 使 用 应 用 程序 时 ， 首 先 启动 客户 端 ， 通 过 有 关 命 令 告 知 服务 器 进行 连接 ， 以 完成 各 
种 操作 ， 而 服务 器 则 按照 此 请 示 提 供 相应 的 服务 。 每 一 个 客户 端 软件 的 实例 都 可 以 向 一 个 服 
务 器 或 应 用 程序 服务 器 发 出 请 求 。 

这 种 系统 的 特点 ， 就 是 客户 端 和 服务 器 程序 不 在 同一 台 计算 机 上 运行 ， 这 些 客户 端 和 服 
务 器 程序 通常 归属 不 同 的 计算 机 。 

主 从 式 架 构 通过 不 同 的 途径 应 用 于 很 多 不 同类 型 的 应 用 程序 。 例 如 ， 现 在 人 们 最 熟悉 的 
在 因特网 上 使 用 的 网 页 。 当 顾客 想 要 在 当当 网 上 买书 的 时 候 ， 电 脑 和 网 页 浏览 器 就 被 当 作 一 
个 客户 端 ， 同 时 ， 组 成 当当 网 的 电脑 、 数 据 库 和 应 用 程序 就 被 当 作 服 务 器 。 当 顾客 的 网 页 浏 
览 器 向 当当 网 请 求 搜寻 数据 库 相 关 的 图 书 时 ， 当 当 网 服务 器 从 当当 网 的 数据 库 中 找 出 所 有 该 
类 型 的 图 书信 息 ， 结 合成 一 个 网 页 ， 再 发 送 回 顾客 的 浏览 器 。 服 务 器 端 一 般 使 用 高 性 能 的 计 
算 机 ， 并 配合 使 用 不 同类 型 的 数据 库 ， 比 如 Oracle、Sybase 或 者 MySQL 等 ， 客 户 端 需要 安 
装 专门 的 软件 ， 比 如 浏览 器 。 


16.1.2 ”数据 库 的 原理 


在 使 用 数据 库 之 前 ， 用 户 必 须 对 数据 库 的 构造 及 运行 方式 有 所 了 解 ， 才 能 有 效 地 制作 互 
动 程序 。 

数据 库 (Database) 是 一 些 相关 数据 的 集合 ， 我 们 可 以 用 一 定 的 原则 与 方法 添加 、 编 辑 和 删除 
数据 的 内 容 ， 进 而 对 所 有 数据 进行 搜索 、 分 析 及 对 比 ， 取 得 可 用 的 信息 ， 产 生 所 需要 的 结果 。 

一 个 数据 库 中 不 是 只 能 保存 一 种 简单 的 数据 ， 可 以 将 不 同 的 数据 内 容 保存 在 同一 个 数据 
库 中 。 例 如 ， 在 进 销 存 管理 系统 中 ， 可 以 同时 将 货物 数据 与 厂商 数据 保存 在 同一 个 数据 库 文 
件 中 ， 归 类 及 管理 时 较为 方便 。 

若 不 同类 的 数据 之 间 有 关联 ， 还 可 以 彼此 使 用 。 例 如 ， 可 以 查询 出 某 种 产品 的 名 称 、 规 
格 及 价格 ， 而 且 可 以 利用 其 厂商 编号 查询 到 厂商 名 称 及 联系 电话 。 我 们 称 保存 在 数据 库 中 不 





同类 别 的 记录 集合 为 数据 表 (Table)， 一 个 数据 库 中 可 以 保存 多 个 数据 表 ， 而 每 个 数据 表 之 间 
并 不 是 互 不 相干 的 ， 如 果 有 关联 的 话 ， 是 可 以 协同 作业 彼此 合作 的 ， 如 图 16-1 所 示 。 


一 个 数据 库 可 以 保存 多 个 不 同 的 
数据 表 ， 科 个 数据 表 之 间 因为 共 
同 拥有 一 个 相同 字段 而 可 以 彼此 


相关 、 协 同 作业 ， 这 就 是 所 谓 的 
“关系 式 数据 库 ” 





图 16-1 数据 库 示 意 
每 一 个 数据 表 都 由 一 个 个 字段 组 合 起 来 。 例 如 ， 在 产品 数据 表 中 ， 可 能 会 有 产品 编号 、 
产品 名 称 、 产 品 价格 等 字段 ， 只 要 按照 一 个 个 字段 的 设置 输入 数据 ， 即 可 完成 一 个 完整 的 数 
据 库 。 如 图 16-2 所 示 。 





日 产品 价格 
产品 a00001 ¥90 
数据 表 








图 16-2 数据 表示 意 


这 里 有 一 个 很 重要 的 概念 ， 一 般 人 认为 数据 库 是 保存 数据 的 地 方 ， 这 是 不 对 的 。 其 实 ， 
数据 表 才 是 真正 保存 数据 的 地 方 ， 数 据 库 是 放置 数据 表 的 场所 。 
16.1.3 MySQL 版 本 

针对 不 同 用 户 ，MySQL 分 为 两 个 不 同 的 版 本 。 

(1) MySQL Community Server( 社 区 版 )。 该 版 本 完全 免费 ， 但 是 官方 不 提供 技术 支持 。 

(2) MySQL Enterprise Server( 企 业 版 服务 器 )。 它 能 够 以 很 高 性 价 比 为 企业 提供 数据 仓库 应 
用 ,支持 ACID 事务 处 理 ， 提 供 完 整 的 提交 、 回 滚 、 骨 溃 恢 复 和 行 级 锁定 功能 。 但 是 该 版 本 
需要 付费 使 用 ， 官 方 提 供电 话 技术 支持 。 

pr MySQL Cluster 主要 用 于 架设 集群 服务 器 ， 需 要 在 社区 版 或 企业 版 基础 上 使 用 。 





MySQL 的 命名 机 制 由 3 个 数字 和 1 个 后 缀 组 成 ， 如 MySQL-5.7.10。 

(1) 第 1 个 数字 (5) 是 主 版 本 号 ， 描 述 了 文件 格式 ， 所 有 版 本 5 的 发 行 版 都 有 相同 的 文件 
格式 。 

(2) 第 2 个 数字 (7) 是 发 行 级 别 ， 主 版 本 号 和 发 行 级 别 组 合 在 一 起 便 构 成 了 发 行 序列 号 。 

(3) 第 3 个 数字 (10) 是 在 此 发 行 系列 的 版 本 号 ， 随 每 次 新 分 发 版 本 递增 。 通 常 选择 已 经 发 
行 的 最 新 版 本 。 

在 MySQL 开发 过 程 中 ， 同 时 存在 多 个 发 布 系列 ， 每 个 发 布 处 在 成 熟 度 的 不 同 阶段 。 

(1) MySQL 5.7 是 最 新 开发 的 稳定 (GA) 发 布 系列 ， 是 将 执行 新 功能 的 系列 ， 目 前 已 经 可 以 
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正常 使 用 。 

(2) MySQL 5.6 是 比较 稳定 (GA) 发 布 系列 。 只 针对 漏洞 修复 重新 发 布 ， 没 有 增加 会 影响 稳 
定性 的 新 功能 。 

(3) MySQL 5.1 是 前 一 稳定 (产品 质量 ) 发 布 系列 。 只 针对 严重 漏洞 修复 和 安全 修复 重新 发 
布 ， 没 有 增加 会 影响 该 系列 的 重要 功能 。 


Sm 对 于 MySQL 4.1、4.0 和 3.23 等 低 于 5.0 的 老 版 本 ， 官 方 将 不 再 提供 支持 。 而 所 
站 ”有 发 布 的 MySQL(Current Generally Available Release) 版 本 已 经 经 过 严格 标准 的 测试 ， 
可 以 保证 其 安全 可 靠 地 使 用 。 针 对 不 同 的 操作 系统 ， 读 者 可 以 在 MySQL 官方 下 载 页 
面 (http://dev.MySQL.com/downloads/) 下 载 到 相应 的 安装 文件 。 


16.1.4 MySQL 的 优势 


MySQL 的 主要 优势 如 下 。 

(1) 速度 。 运 行 速度 快 。 

(2) 价格 。MySQL 对 多 数 个 人 用 户 来 说 是 免费 的 。 

(3) 容易 使 用 。 与 其 他 大 型 数据 库 的 设置 和 管理 相 比 ， 其 复杂 程度 较 低 ， 易 于 学 习 。 

(4) 可 移植 性 。 能 够 工作 在 众多 不 同 的 系统 平台 上 ， 如 Windows、Linux、UNIX、Mac 
OS 等 操作 系统 。 

(5) 丰富 的 接口 。 提 供 了 用 于 C、C++、Eiffel、Java、Perl、PHP、Python、Ruby 和 Tcl 
等 语言 的 API。 

(6) 支持 查询 语言 。MySQL 可 以 利用 标准 SQL 语法 并 且 支 持 ODBC( 开 放 式 数据 库 连 接 ) 
的 应 用 程序 。 

(7) 安全 性 和 连接 性 。 十 分 灵活 和 安全 的 权限 和 密码 系统 ， 人 允许 基于 主机 的 验证 。 连 接 
到 服务 器 时 ， 所 有 的 密码 传输 均 采 用 加 密 形式 ， 从 而 保证 了 密码 安全 。 并 且 由 于 MySQL 是 
网 络 化 的 ， 因 此 可 以 在 因特网 上 的 任何 地 方 访问 ， 提 高 了 数据 共享 的 效率 。 


16.2 ”创建 MySQL 数据 库 和 数据 表 


MySQL 数据 库 的 指令 都 是 在 命令 提示 符 界 面 中 使 用 的 。 但 这 对 于 初学 者 是 比较 困难 的 。 
针对 这 一 难题 ， 本 书 将 采用 phpMyAdmin 管理 程序 来 执行 ， 以 便 能 有 更 简易 的 操作 环境 与 使 
用 效果 。 


16.2.1 启动 phpMyAdmin 管理 程序 


phpMyAdmin 是 一 套 使 用 PHP 程序 语言 开发 的 管理 程序 ， 它 采用 网 页 形式 的 管理 界面 。 
如 果 要 正确 执行 这 个 管理 程序 ， 就 必须 要 在 网 站 服务 器 上 安装 PHP 与 MySQL 数据 库 。 
EC 如 果 要 启动 phbpMyAdmin 管理 程序 ， 只 要 单 击 桌 面 右 下 角 的 WampServer 图 
标 ， 在 弹出 的 菜单 中 选择 phpMyAdmin 命令 ， 如 图 16-3 所 示 。 


EEJppphpMyAdmin 启动 后 进入 登录 页 面 。 在 默认 情况 下 ，MySQL 数据 库 的 管理 员 用 
户 名 为 root， 密 码 为 空 。 输 入 完成 后 ， 单 击 “ 执 行 ” 按 钮 ， 如 图 16-4 所 示 。 


Made in France by Otomatig 





FE Your VirtualHo 拖 
思 www 目录 (W) 
FE Apache 


WAMPSERVER 3.0.6 





图 16-3 选择 phpMyAdmin 命令 


EEJS) 进入 phpMyAdmin 的 工作 界面 ， 如 图 16-5 所 示 。 用 户 只 需要 单 
接 ， 即 可 创建 新 的 数据 库 。 








口 x 
加 - @ hepi/Mocalhoswphprmyadmin/server detal P - ©| @ localhost / Local Databa.. x 
文件 (月 ” 饥 铭 {人 E) 到 看 (V) 收藏 实 (A) 工具 ( (+) 
phpMyAdmin 上 
全 加 间 门 泰信 j 数 氏 库 加 SOL 总 关 5 2 里 P 司 届 区 BA vs 
近 源 访问 ”去 履 孝 二 
Be 本 者 应 规则 拒 作 
er infomaton schema utts_general oi ss 检查 权限 
各 加 Raton_ scheme sg erlnz_swedlsn_c1 办 吾 权限 
sal 
i periormance_schama utt8_general_ci a 容 查 权限 
店 司 ss oj rse_qeneral_=i *- 的 百 权限 
Sit:4 Jatim_swedish_o 
tt 口 训 大 # 硕 :已 昌 除 
注意 : 在 此 富有 数据 幸 计 可 能 导致 网站 豫 务 基 和 MySQL 服务 对 之 同 的 流 县 玖 增 
* 启用 统计 
-wv 
my 
httpxtlocalhostfphpmyadminy/server_databasesPhp?server=1atoken=465140aDdczab51b019c483fa525c139 起 100% vv 
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图 16-4 数据库 登 录 页 面 

























16.2.2 ”创建 数据 库 
在 MySQL 数据 库 中 ， 


performance_ schema 和 sys。 


会 有 4 个 内 置 数据 库 : 


图 16-5 phpMyAdmin 的 工作 界面 





“新 建 ” 链 


册 满 内 1DOSAW 窒 注 UWpyAWdyd 一 一 丑 兰 号 19SAW 消 咕 ”机 91 中 鱼 





mysql 、 information_schema 、 


(1) mysql 数据 库 是 系统 数据 库 ， 在 24 个 数据 表 中 保存 了 整个 数据 库 的 系统 设置 ， 十 分 重要 。 
(2) information schema 包括 数据 库 系统 有 什么 库 ， 有 什么 表 ， 有 什么 字典 ， 有 什么 存储 
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过 程 等 所 有 对 象 信息 和 进程 访问 、 状 态 信息 。 
(3) performance schema 新 增 一 个 存储 引擎 ， 主 要 用 于 收集 数据 库 服务 器 性 能 参数 。 包 括 
锁 、 互 斥 变量 、 文 件 信息 ;保存 历史 的 事件 汇总 信息 ， 为 提供 MySQL 服务 器 性 能 做 出 详细 的 
判断 ， 对 于 新 增 和 删除 监控 事件 点 都 非常 容易 ， 并 可 以 随意 改变 MySQL 服务 器 的 监控 周期 。 
(4) sys 数据 库 为 系统 数据 库 ， 通 过 这 个 数据 库 可 以 快速 地 了 解 系统 的 元 数据 信息 。 


sys 数据 库 是 MySQL 5.7 新 增 的 一 个 功能 ， 这 个 数据 库 是 通过 视图 的 形式 把 
information schema 和 performance_ schema 结合 起 来 ， 查 询 出 令 人 更 加 容易 理解 的 数据 。 

这 里 以 在 MySQL 中 创建 一 个 企业 员工 管理 数据 库 company 为 例 ， 并 添加 一 个 员工 信息 
表 employee。 如 图 16-6 所 示 ， 在 文本 框 中 输入 要 创建 数据 库 的 名 称 company， 再 单 击 “ 创 
建 ” 按 钮 即 可 。 


“um 
了 提示 





[companyl x | | 排序 规则 癌 @ 


16-6 创建 数据 库 company 


Sen 在 一 个 数据 库 中 可 以 保存 多 个 数据 表 ， 以 本 页 所 举 的 范例 来 说 明 : 一 个 企业 员工 管 


AS 理 的 数据 库 中 ， 可 以 包含 员工 信息 数据 表 、 岗 位 工资 数据 表 、 销 售 业绩 数据 表 等 。 因 
此 ， 这 里 需要 创建 数据 库 company， 也 需要 创建 数据 表 employee。 


16.2.3 ”认识 数据 表 的 字段 


在 添加 数据 表 之 前 ， 首 先 要 规划 数据 表 中 要 使 用 的 字段 。 其 中 设置 数据 字段 的 类 型 非常 
重要 ， 使 用 正确 的 数据 类 型 才能 正确 地 保存 和 应 用 数据 。 

在 MySQL 数据 表 中 常用 的 字段 数据 类 型 可 以 分 为 以 下 3 个 类 别 。 

1. 数值 类 型 


可 用 来 保存 、 计 算 的 数值 数据 字段 ， 如 会 员 编号 或 是 产品 价格 等 。 在 MySQL 中 的 数值 
字段 按照 保存 的 数据 所 需 空间 大 小 有 以 下 区 别 ， 如 表 16-1 所 示 。 











表 16-1 数值 类 型 
数值 数据 类 型 保存 空间 数据 的 表示 范围 
TINYINT signed -128~ 127/unsigned 0~255 
SMALLINT signed-32768 一 32767/unsigned0 一 65 535 
MEDIUMINT Signed-8388 608 一 8388 607hnsigned0 一 16777215 
INT Signed-2147483 648 一 2 147 483 647/unsigned0 一 4294967295 





注 : signed 表示 其 数值 数据 范围 可 能 有 负 值 ，unsigned 表示 其 数值 数据 均 为 正 值 。 


2. 日 期 及 时 间 类 型 


可 用 来 保存 日 期 或 时 间 类 型 的 数据 ， 如 会 员 生 日 、 留 言 时 间 等 。MySQL 中 的 日 期 及 时 间 
类 型 有 下 列 几 种 格式 ， 如 表 16-2 一 表 16-4 所 示 。 


表 16-2 日 期 数据 类 型 








数据 类 型 名 称 DATE 

存储 空间 3 

数据 的 表示 范围 "1000-01-01" 一 '9999-12-31" 

数据 格式 "YYYY-MM-DD" "YY-MM-DD" "YYYYMMDD" "YYMMDD" 





YYYYMMDD YYMMDD 
注 : 在 数据 格式 中 ， 若 没有 加 上 引号 则 为 数值 的 表示 格式 ; 若 前 后 加 上 引号 则 为 字符 串 的 表示 格式 。 


表 16-3 时间 数 据 类 型 


数据 类 型 名 称 
存储 空间 
'-838:59:59'—'838:59:59" 





"hhmmss" hhmmss 
注 : 在 数据 格式 中 ， 若 没有 加 上 引号 则 为 数值 的 表示 格式 ， 若 前 后 加 上 引号 则 为 字符 串 的 表示 格式 。 


表 16-4 日 期 与 时 间 数 据 类 型 


数据 类 型 名 称 DATETIME 

存储 空间 8 byte 

数据 的 表示 范围 '1000-01-01 00:00:00'~'9999-12-31 23:59:59' 

数据 格式 "YYYY-MM-DD hh:mm:ss” "YY-MM-DD hh:mm:ss” "YYYYMMDDhhmmss" 


"YYMMDDhhmmss" YYYYMMDDhhmmss YYMMDDhhmmss 
注 : 在 数据 格式 中 ， 若 没有 加 上 引号 则 为 数值 的 表示 格式 ， 若 前 后 加 上 引号 则 为 字符 串 的 表示 格式 。 


3. 文本 类 型 
可 用 来 保存 文本 类 型 的 数据 ， 如 学 生 姓名 、 地 址 等 。 在 MySQL 中 文本 类 型 数据 有 下 列 
几 种 格式 ， 如 表 16-5 所 示 。 





表 16-5 文本 数据 类 型 










文本 数据 类 型 保存 空间 数据 的 特性 

CHAR(M) M bytes， 最 大 为 255 bytes 必须 指定 字段 大 小 ， 数 据 不 足 时 以 空白 字符 填 满 
M bytes， 最 大 为 255 bytes ”| 必须 指定 字段 大 小 ， 以 实际 填 入 的 数据 内 容 来 存储 

最 多 可 保存 25 535 bytes 不 需 指定 字段 大 小 


在 设置 数据 表 时 ， 除 了 要 根据 不 同性 质 的 数据 选择 适合 的 字段 类 型 之 外 ， 有 些 重要 的 字 
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段 特 性 定义 也 能 在 不 同 的 类 型 字段 中 发 挥 其 功能 ， 常 用 的 设置 如 下 ， 如 表 16-6 所 示 。 
表 16-6 ”特殊 字段 数据 类 型 


定义 内 容 
定义 数值 数据 中 是 否 允 许 有 负 值 ，SIGNED 表示 人 允许 
自动 编号 ， 由 0 开始 以 1 来 累加 


特性 定义 名 称 适用 类 型 
SIGNED.UNSIGNED | 数值 类 型 
AUTOJNCREMENT 数值 类 型 




















BINARY 文本 类 型 保存 的 字符 有 大 小 写 区 别 
NULL.NOTNULL | 全 部 是 否 允 许 在 字段 中 不 填 入 数据 
默认 值 全 部 若是 字段 中 没有 数据 ， 即 以 默认 值 填充 
主妇 全 部 主 索引 ， 每 个 数据 表 中 只 能 允许 一 个 主键 列 ， 而 且 该 栏 数据 
不 能 重复 ， 加 强 数据 表 的 检索 功能 
16.2.4 创建 数据 表 


要 添加 一 个 员工 信息 数据 表 ， 如 表 16-7 所 示 是 这 个 数据 表 字 段 的 规划 。 
表 16-7 员工 信息 数据 





名 称 字 自 是 否 为 空 
员工 编号 cmD 否 
姓名 cmName 否 
性 别 cmsex 否 
生日 cmBirthda 否 
电子 邮件 cmEmail 是 
电话 cmPhone 是 
住址 cmAddress 是 
其 中 有 以 下 几 个 要 注意 的 地 方 。 


(1) 员工 编号 (cmID) 为 这 个 数据 表 的 主 索引 字段 ， 基 本 上 它 是 数值 类 型 保存 的 数据 ， 因 为 
一 般 座 号 不 会 超过 两 位 数 ， 也 不 可 能 为 负数 ， 所 以 设置 它 的 字段 类 型 为 TINYINT(2)， 属 性 为 
UNSIGNED。 在 添加 数据 时 ， 数 据 库 能 自动 为 学 生 编号 ， 所 以 在 字段 上 加 入 了 auto_increment 


自动 编号 的 特性 。 

(2) 姓名 (cmName) 属 于 文本 字段 ， 一 般 不 会 超过 10 个 中 文字 ， 也 就 是 不 会 超过 20 Bytes， 
所 以 这 里 设置 为 VARCHAR(20)。 

(3) 性 别 (cmSex) 属 于 文本 字段 ， 因 为 只 保存 一 个 中 文字 ( 男 或 女 )， 所 以 设置 为 CHAR(2)， 
默认 值 为 “ 男 ”。 


(4) 生日 (cmBirthday) 属 于 日 期 时 间 格式 ， 设 置 为 DATE。 
(5) 电子 邮件 (cmEmaiD 和 住址 (cmAddress) 都 是 文本 字段 ， 设 置 为 VARCHAR(100)， 最 多 
可 保存 100 个 英文 字符 ，50 个 中 文字 。 电 话 (cmPhone) 设 置 为 VARCHAR(100)。 因 为 每 个 人 
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不 一 定 有 这 些 数据 ， 所 以 这 3 个 字段 允许 为 空 。 
接着 就 要 回 到 phpMyAdmin 的 管理 界面 ， 为 MySQL 中 的 company 数据 库 添 加 数据 表 。 

在 左 侧 列表 中 选择 创建 的 company 数据 库 ， 输 入 添加 的 数据 表 名 称 和 字段 数 ， 然 后 单 击 “ 执 

行 ” 按 钮 ， 如 图 16-7 所 示 。 


@ [| rep//ocalhosohpryadnin/sever doabae: P - © | @ ocaltont / Local Detaba— yx 介 妆 窜 @ 
文人 中 ”多 和 日。 下 看 M =。 工具 mn) 
phpMyAdmin 
会 国 基 可 堆 号 
访问 雪 收 区 
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后 Er 
拒 100% 





图 16-7 新建 数据 表 employee 
如 图 16-8 所 示 为 添加 的 数据 表 字 段 ， 请 按照 其 中 的 内 容 设置 数据 表 。 
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数据 吉 名 :employee 添加 [1 | 个 字 掺 (仿生 
畏 构 号 
名 字 类 型 各 长 魔 /| 值 已 黑 认 电 排序 规则 属性 空 
emiD | [wr 图 fs 无 国 | |gb2312_chinese 图 | |wsexeo 国 | 固 
Fry | IVARcHAR ” 国 fan [天 加 | | sz2312_chnese 回 | [加 | 
cmSex | [ee 国 EE 定义 国 [e2312 chinese @]| | 贺 日 
男 

cmBirhday nl DATE 国 | | | 无 四 | | gb2312_chinese 四 | | 国 | 
cmEmail [ARcHAR 国 | [im 无 梧 | | %2312_chinese 区 画 | 图 
cmPhone | [|wRchAR 区 匠 | 无 国 | | sz2312_ cnese 回 | | _ 国 | 加 
omAddress [WARchAR _ 国 | [fio 无 国 [e2312_chinese 回 国 图 





























图 16-8 ”添加 数据 表 字段 

设置 的 过 程 中 要 注意 以 下 4 点 。 

(1) 设置 cmID 为 整数 。 

(2) 设置 cmID 为 自动 编号 。 

(3) 设置 cmID 为 主键 列 。 

(4) 允许 cmEmail、cmPhoned、cmAddress 为 空位 。 

在 设置 完毕 之 后 ， 单 击 “ 保 存 ” 按 钮 ， 在 打开 的 界面 中 可 以 查看 完成 的 employee 数据 
表 ， 如 图 16-9 所 示 。 
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[ 华 服 务 器 : mysql Et company 去 -employee 
园 浏览 3 结构 司 SQL 和 搜索 了 插入 忆 SU 吏 导 入 号 权限 咏 操 作 三 名 发 器 
并 名 字 类 型 排序 规则 属性 空 默认 格外 操作 
1 cmlD int(8) WMD 天 无 妇 修 改 @ 出 除 今 主键 国 唯一 圈 未 引 阳 空间 vv 更 多 
2 cmName varchar(20) gb2312 chinese_ci 否 无 修改 合 出 除 沪 主 键 司 唯一 司 索 引 卫 空间 vw 更 多 
3 cmSex char(2) gb2312_chinese_ci 否 男 姥 修 改 全 删除 户主 键 国 惟一 厨 索 引 卫 空间 v 更 多 
4 cmBirthday date 普 无 人 让 改 全 删除 思 主 键 而 唯一 司 索引 长 空 间 立 更 多 
5 cmEmail varchar(100) gb2312_chinese_ci 是 NULL 妇 修 改 四 出 际 户主 键 国 瞧 一 辐 索 引 阳 空间 vv 更 多 
] 6 cmPhone varchar(50) gb2312_chinese_ci 是 NULL 这 修 沁 全 删除 力主 键 恒 唯 一 司 索引 并 空间 v 更 多 
7 cmAddress varchar(100) gb2312_chinese_ci 是 NULL 光 修 收 @ 唱 睹 户主 计 国 叭 一同 过 引 阳 空 间 vv 更 多 


人 回 全 选 ” 进 刺 国 浏 览 只 修改 四 出 除 户主 六 丽 唯 - 司 案 引 








6-9 employee 数据 表 





16.2.5 ”添加 数据 


添加 数据 表 后 ， 还 需要 添加 具体 的 数据 。 具 体操 作 步 又 如 下 。 
EC 选择 employee 数据 表 ， 选 择 菜单 上 的 “插入 ”链接 。 依 照 字段 的 顺序 ， 将 对 应 
的 数值 依次 输入 ， 单 击 “ 执 行 ”按钮 ， 即 可 插入 数据 ， 如 图 16-10 所 示 。 


[--] [国有 服务 器 ; mysql wampserver EE :> 旧 去 : employee 





辣 浏览 jh 结构 司 SQL 导热 索 了 扩 括 和 A 加 号 局 导入 中 权限 让 操作 污 扣发 器 
字段 类 型 国教 空 值 
cmlD int(8) unsigned 


10001 
cmName 。 varcharf20) 





| 王 艺 
cmSex char(2) 国 页 
Birthday date 加 | E 
A el | 19820602 | 回 
cmEmail varchar(100) 国 | 回 pingguo8163.ccm < 


re rp 国 | 回 fo992-1234567 














cmAddress varchar(100) | 加 | 加 长 吗 路 12 引 “ 
a 
图 16-10 插入 数据 
EEJ2y 按照 图 16-11 所 示 的 数据 ， 重 复 执行 上 一 步 的 操作 ， 将 数据 输入 到 数据 表 中 。 
cmlD cmName cmSex cmBirthday cmEmail cmPhone cmAddress 


10001 王猛。 | 男 |1982-06-02 pingguo@163.com “0992-1234567 | 长 鸣 路 12 号 
10002 王 小 敏 。 女 。 “1972-06-02 wangxiaomin@163 .com 0992-1234560 | 西 华 街 19 号 
10003 张 华 。 男 |1970-06-02 zhanghua@163.com ”0992-1234561 | 长安 路 20 号 
10004 王菲 女 1982-03.02 wangfei@163.com 0992-1234562 兴隆 街 11 号 
10005 杨康 男 |1978-06-02 yangkang@163.com ”0992-1234568 | 长 安 街 20 号 
10006 冯 菲 菲 ” 女 。 ”1982-03-20 fengfeifei@163 com “0992-1234512 长安 街 42 号 


16-11 输入 的 数据 


16.3 ”加密 MySQL 数据 库 


下 面 介绍 MySQL 数据 库 的 高 级 应 用 ， 主 要 包括 MySQL 数据 库 的 安全 、MySQL 数据 库 
的 加 密 等 内 容 。 


16.3.1 _ MySQL 数据 库 的 安全 问题 


MySQL 数据 库 是 存在 于 网 络 上 的 数据 库 系 统 。 只 要 是 网 络 用 户 ， 都 可 以 连接 到 这 个 资 
源 。 如 果 没 有 权限 或 其 他 措施 ， 任 何人 都 可 以 对 MySQL 数据 库 进行 存 取 。MySQL 数据 库 在 
安装 完毕 后 ， 默 认 是 完全 不 设防 的 。 也 就 是 任何 人 都 可 以 不 使 用 密码 就 连接 到 MySQL 数据 
库 。 这 是 一 个 相当 危险 的 安全 漏洞 。 

1. phpMyAdmin 管理 程序 的 安全 考虑 

phpMyAdmin 是 一 套 网 页 界面 的 MySQL 管理 程序 。 有 许多 PHP 的 程序 设计 师 都 会 将 这 
套 工具 直接 上 传 到 他 的 PHP 网 站 文件 夹 里 。 管 理 员 只 能 从 远 端 通过 浏览 器 登录 phpMyAdmin 
来 管理 数据 库 。 

这 个 方便 的 管理 工具 是 否 也 是 方便 的 入 侵 工具 呢 ? 没 错 。 只 要 是 对 phpMyAdmin 管理 较 
为 熟悉 的 朋友 ， 看 到 该 网 站 是 使 用 PHP+MySQL 的 互动 架构 ， 都 会 去 测试 该 网 站 
<phpMyAdmin> 的 文件 夹 是 否 安装 了 phpMyAdmin 管理 程序 。 若 是 网 站 管理 员 一 时 朴 忽 ， 很 
容易 让 人 猜 中 ， 进 入 该 网 站 的 数据 库 。 


2. 防 堵 安全 漏洞 的 建议 


无 论 是 MySQL 数据 库 本 身 的 权限 设置 ， 还 是 phpMyAdmin 管理 程序 的 安全 漏洞 ， 为 了 
避免 他 人 通过 网 络 入 侵 数 据 库 ， 必 须要 先 做 以 下 几 件 事 。 

(1) 修改 phpMyAdmin 管理 程序 的 文件 夹 名 称 。 这 个 做 法 虽然 简单 ， 但 至 少 已 经 挡 掉 一 
大 半 非 法 入 侵 者 了 。 最 好 是 修改 成 不 容易 猜 到 并 且 与 管理 或 是 MySQL、phpMyAdmin 等 关键 
字 无 关 的 文件 夹 名 称 。 

(2) 为 MySQL 数据 库 的 管理 账号 加 上 密码 。 我 们 一 再 提 到 MySQL 数据 库 的 管理 账号 
root， 默 认 是 不 设 任何 密码 的 。 这 就 好 像 装 了 安全 系统 ， 却 没 打开 电源 开关 一 样 ， 所 以 替 root 
加 上 密码 是 相当 重要 的 。 

(3) 养 成 备份 MySQL 数据 库 的 习惯 。 当 用 户 一 旦 所 有 安全 措施 都 失效 了 ， 若 平常 就 有 备 
份 的 习惯 ， 即 使 数据 被 删除 了 ， 还 能 很 轻松 地 恢复 。 


16.3.2 为 MySQL 管理 账号 加 上 密码 


在 MySQL 数据 库 中 的 管理 员 账 号 为 root， 为 了 保护 数据 库 账 号 的 安全 ， 我 们 可 以 为 管理 
员 账 号 加 密 。 具 体操 作 步 又 如 下 。 
EC 进入 phpMyAdmin 的 管理 主 界面 。 单 击 “ 权 限 ” 链 接 ， 来 设置 管理 员 账 号 的 权 
限 ， 如 图 16-12 所 示 。 
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喇 服 务 器 : mysql wampserver » 苹 数据 库 - company 








缉 结 和 构 加 SQL 忌 搜 索 回 坦 询 居 S 出 司 叶 访 提 作 oa 更 多 
表 。 操作 行 数 类 型 ”排序 ; 大 条 
志 小 余 
employee 国 浏 览 驻 结 构 合 搜索 也 插入 狗 清空 加 出 除 ~6 InnoDB latin1_ swedish ci 16 - 

KB 
1 张 表 。 总 计 6 InnoDB latin1_swedish ci 16 0 
至 字 
节 

企 _ 回 全 选 选中 项 : 二 














16-12 ”设置 管理 员 密码 


ER 这 里 有 两 个 root 账号 ， 分 别 为 由 本 机 (localhost) 进 入 和 所 有 主机 (: : 1) 进 入 的 
管理 账号 ， 默 认 没有 密码 。 首 先 修改 所 有 主机 的 密码 ， 单 击 “ 编 辑 权 限 ” 链 接 ， 如 
图 16-13 所 示 。 进 入 下 一 页 。 

一 EE 


有 结构 司 SQL 导 执 索 可 查询 加 导出 辐 呈 六 所 作 a: 权限 vv 更 多 








」 驴 用 户 可 以 访问 "company” 上 


用 户 主机 类 型 权限 授权 损 作 
root 127.0.0.1 全 局 ALL PRIVILEGES 是 ”局 总 辑 权限 
root 21 全 局 ALL PRIVILEGES 是 ”已 编辑 权限 


root localhost 全 局 ALL PRIVILEGES 是 属 编 辑 权限 


| 


总 添加 用 户 


图 16-13 ”查看 用 户 界面 


ECDIS 在 打开 的 界面 中 的 “密码 ”文本 框 中 输入 所 要 使 用 的 密码 ， 如 图 16-14 所 示 。 单 
击 “ 执 行 ”按钮 ， 即 可 添加 密码 。 








| 个 光 灾 妈 | 
目 天 这 如 
同 密码 : | 重新 输入 : | 
密码 加 志方 式 ; 加 MySQL4.1+ 
日 MySQL40 莱 容 


生成 更 码 GE 








16-14 ”添加 密码 


16.4 数据 库 的 备份 与 还 原 


在 MySQL 数据 库 里 备份 数据 是 十 分 简单 又 轻松 的 事情 。 下 面 介绍 如 何 备份 MySQL 的 数 
据 表 ， 以 及 数据 表 的 删除 与 插入 操作 。 


16.4.1 数据库 的 备份 


用 户 可 以 使 用 phpMyAdmin 的 管理 程序 将 数据 库 中 的 所 有 数据 表 导 出 成 一 个 单独 的 文本 
文件 。 当 数据 库 受 到 损坏 或 是 要 在 新 的 MySQL 数据 库 中 加 入 这 些 数据 时 ， 只 要 将 这 个 文本 
文件 插入 即 可 。 

以 本 章 所 使 用 的 文件 为 例 ， 先 进入 phpMyAdmin 的 管理 界面 ， 下 面 就 可 以 备份 数据 库 
了 。 有 具体 操作 步骤 如 下 。 

EEC 选择 需要 导出 的 数据 库 ， 单 击 “ 导 出 ”链接 ， 选 择 导 出 方式 为 “快速 -显示 最 少 

的 选项 ”， 单 击 “ 执 行 ” 按 钮 ， 如 图 16-15 所 示 。 
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oe 二 口 x 
[ 国 -|@ rpy/ocalhosyphpmyadmin/db exportph P ~ ©|| @ lccalhost / Local Databa.. x A 
文件 (有 。 疯 辑 (E) ”至 看 (V) 收藏 区 (A) 工具 (m) 
phpMyAdmin -Es mpany 周 
会 国 日 可 窜 $ 和 rw 结构 加 SQL 又 搜索 日 查 9 加 SB 出 加 SA 更 多 
近 基 访问 表 收 藏 夫 
ee | 
3 正在 导出 数据 库 “company" 中 的 数据 表 
company 
bm 3 information_schema 时 由 方式 : 
日 回 mysql 
南 可 perormance_schema 图 快速 - 号 示 最 少 的 选项 
Easys 〇 让 定义 -显示 所 有 可 用 的 选项 
a mi 
用 洛 六 Miz 树 格式 
IE 
国 辐 要 
局 新 建 SQL 可 
鲍 W sys_config 
由 国 规 四 
v 
转 100% ~ 











EEC 打开 “另存 为 ”对 话 框 ， 在 其 中 输入 保存 文件 的 名 称 ， 设 置 保存 的 类 型 及 位 


置 ， 如 图 16-16 所 示 。 


pre MySQL 备份 下 的 文件 是 扩展 名 为 *.sql 的 文本 文件 ， 这 样 的 备份 操作 不 仅 简单 ， 


也 提示 
SS 文件 内 容 也 较 小 。 


16-15 ”选择 要 导出 的 数据 库 
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没有 与 搜索 冬 件 开 配 的 项 - 


呈 


























图 16-16 “另存 为 ”对 话 框 
16.4.2 ”数据 库 的 还 原 


还 原 数据 库 文件 的 具体 操作 步骤 如 下 。 

在 执行 数据 库 的 还 原 前 ， 必 须 将 原来 的 数据 表 删 除 。 单 击 employees 数据 表 右 侧 
的 “删除 ”链接 ， 如 图 16-17 所 示 。 

EEJRD9 此 时 会 显示 一 个 询问 画面 ， 单 击 “确定 ”按钮 ， 如 图 16-18 所 示 。 

[1 服务 器 : mysql wampserver » 煌 数据 原 . company 


号 结构 让 SQL 起 拓 索 品 查 询 加 SL 加 SA 7 
表 ~ 操作 行 数 








地 x 
口 employee ，] 浏 览 结构 过 搜索 比 抵 入 全 青空 ens 2 您 将 要 完全 删除 一 个 表 ! 您 真 的 要 执行 "DROP 


lis | TABLE employee" 吗 ? 
1 张 表 总 计 6 


人 回 全 先 [站 项: 加 取消 , 
图 16-17 单 击 “ 删 除 ” 链 接 图 16-18 ”信息 提示 框 


EEB9 回 到 原 界 面 ， 会 发 觉 该 数据 表 已 经 被 删除 了 ， 如 图 16-19 所 示 。 
ES 接着 要 插入 刚才 备份 的 <company.sql> 文 件 ， 将 该 数据 表 还 原 。 单 击 “ 导 入 ” 链 
接 ， 打 开 “ 要 导入 的 文件 ”界面 ， 如 图 16-20 所 示 。 














EEEETEZTTZTTTTITI 





访 仅 物 加 SQL 飞 扫 二 人 本 Sk 国 SN 小 四 


— “加 服 务 器 : mysql wampserver ， 图 数 据 库 : company 








导入 到 数据 库 "company” 
纪 结构 名 SQL 及 搜索 回 查 得 尼 Ls 加 导入 




















表 。 操作 行 数 包 类 型 ”排序 规则 大 小 ”多余 A 
0 张 表 总 计 0 InnoDB latin1_swedish_ci 0 字 节 0 字 节 Bt ni 
入 讨 且 机 中 上 全 冯 大 | (各 大 限制 : 128 MB) 
企 _ 回 全 迁 选中 项 : = | 国 
16-19 已 经 删除 数据 表 图 16-20 “要 导入 的 文件 ”界面 


EERDp 单 击 界面 中 的 “浏览 ”按钮 ， 打 开 “ 选 择 要 加 载 的 文件 ”对 话 框 ， 选 择 文本 文 
件 company.sql， 单 击 “ 打 开 ” 按 钮 ， 如 图 16-21 所 示 。 






































局 过 要 0 的 文件 i 
< * 个 量 ，H 随 ，T 或 Y 局 笃 雪 -下载 - 户 
组 可 =。 新建 件 赤 -me 
并 忆 访 上 。。 向 各 一 修改 日 3 
国 吉本 w L [DD company.sql 2017/8/1417:44 。 SQL 文件 
4 + 
图 阔 小 有 
EL +* 
| ch13 
ch14 
nm YY < > 
文件 各 (N): [company.sql ~| [FR -| 
Cm |] 








图 16-21 “选择 要 加 载 的 文件 ”对 话 框 


TY 单 击 “ 执 行 ” 按 钮 ， 系 统 即 会 读 取 company.sql 文件 中 所 记录 的 指令 与 数据 ， 将 
数据 表 恢 复 ， 如 图 16-22 所 示 。 


导入 到 数据 库 “company” 


要 导入 的 文件 ; 


这 件 可 用 区 失 gz bop2_ zp) 肥 二 坟 。 
i 


以 计算 机 中 上 但: 。 UserAdmnieatorU EREEE (号 大 由利 : 128 MB) 
六 伯 的 字符 逢 。 | 司 





部 分 导入 : 


国 在 号 入 时 用 丰 和 必 到 可 能 闪 于 区 费 红 长 时 间 《接近 PHP 起 时 的 限定 ) 刚 多 许 中 新 * (他 过 会 闻 村 和 不 ， 安 此 呈 人 太 
ET] 


从 基 一 个 开 击 四 过 有 吉本 执 ‘SQL 有 N) 或 行 对 (其他 网 ) ;| | 








ft: 
[so 百 
格式 特定 过: 
saL 半 :|NONE 。。 国 
国 不 要 给 要 十 余 用 自 央 (aro_TpcRzMEYT) 
和 


图 16-22 开始 执行 导入 操作 


国 于 7 在 执行 完毕 后 ，company 数据 库 中 又 出 现 了 一 个 数据 表 employee， 如 图 16-23 
所 示 。 


mysql wamps 





J 十 后 SQL 广 扫 索 回 坦 询 局 S 加 SA 六 雪人 ff 沾 和 了 从 和 Fs 1 
表 损 作 _ 行 数 馈 类 型 ”排序 规则 大 小 


回 employee 加 浏览 牟 站 构 仿 搜索 了 括 入 锯 青空 @ 出 除 -6 InnoDB latin_swedish ci 16 KB 
1 张 表 入 6 InnoDB latin1_swedish_ci 16 EB 
1 Ds [Rm 下 




















16-23 ”导入 的 数据 表 
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16.5 疑难 解 惑 


疑问 1: 预览 网 页 时 提示 如 图 16-24 所 示 的 错误 信息 ， 如 何 解 决 ? 








HW"|@ htpi//localhost/code/chis/161.php PD ~ ©| @localhost 
文件 (和 ” 妨 铝 (E) 。 坦 看 (V) 收 襄 夫 (A) 工具 (T) 帮助 (H) 








LLL ooo5 | 342872]{main}() [-\16 1 php:0 








图 16-24 ”错误 信息 
答 : 出 现 上 面 的 错误 信息 ， 主 要 是 因为 从 MySQL 7 开始 ， 提 示 用 户 mysql_connect 这 个 
模块 已 经 弃 用 ， 这 里 需要 使 用 mysqli 或 者 PDO 来 蔡 代 。 
将 类 似 以 下 连接 语句 : 


$link = mysql_connect ('localhost', 'user', 'password'); 
mysql_select_db('dbname', $1link); 


修改 如 下 : 


$link = mysqli_connect('localhost', 'user', 'password', '‘'dbname'); 


疑问 2: 如 何 导出 制定 的 数据 表 ? 
答 : 如 果 用 户 想 导出 制定 的 数据 表 ， 在 选择 导出 方式 时 ， 选 中 “ 自 定义 -显示 所 有 可 用 的 
选项 ” 单 选 按钮 ， 然 后 在 “数据 表 ” 列 表 中 选择 需要 导出 的 数据 表 即 可 ， 如 图 16-25 所 示 。 
导出 方式 : 


日 快速 - 显示 最 少 的 选项 
图 | 自 定义 - 显示 所 有 可 用 的 选项 





数据 表 : 


全 选 / 全 不 选 





16-25 ”设置 导出 方式 
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第 17 章 
数据 库 编程 一 一 
MySQL 数据 库 

与 SQL 查询 


如 果 想 更 加 深入 地 使 用 MySQL 数据 库 ， 就 需要 进一步 学 习 MySQL 中 相关 的 
SQL 语句 。 本 章 讲述 MySQL 5.7 数据 库 如 何 独立 安装 和 配置 、MySQL 服务 器 上 的 
重要 操作 等 知识 。 
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17.1 安装 与 配置 MySQL 5.7 
Windows 平台 下 安装 MySQL， 可 以 使 用 图 形 化 的 安装 包 。 图 形 化 的 安装 包 提供 了 详细 的 
安装 向 导 ， 通 过 向 导 ， 读 者 可 以 一 步 一 步 地 完成 对 MySQL 的 安装 。 下 面 介绍 使 用 图 形 化 安 
装 包 安 装 MySQL 的 步骤 。 


17.1.1 安装 MySQL 5.7 


NN 要 想 在 Windows 中 运行 MySQL， 需 要 32 位 或 64 位 Windows 操作 系统 ， 如 Windows 





7、Windows 8、Windows 10、Windows Server 2003、Windows Server 2008 等 。Windows 可 以 
将 MySQL 服务 器 作为 服务 来 运行 。 通 常 ， 在 安装 时 需要 具有 系统 的 管理 员 权 限 。 

Windows 平台 下 提供 两 种 安装 方式 : MySQL 二进制 分 发 版 (msi 安装 文件 ) 和 免 安装 版 
(.zip 压缩 文件 )。 一 般 来 讲 ， 应 当 使 用 二 进 制 分 发 版 ， 因 为 该 版 本 比 其 他 的 分 发 版 使 用 起 来 要 
简单 ， 不 再 需要 其 他 工具 来 启动 就 可 以 运行 MySQL。 这 里 ， 在 XP 平台 上 选用 图 形 化 的 二 进 
制 安装 方式 ， 其 他 Windows 平台 上 安装 过 程 也 差不多 。 


1. 下 载 MySQL 安装 文件 





下 载 MySQL 安装 文件 的 具体 操作 步骤 如 下 。 
打开 正 浏览 器 ， 在 地 址 栏 中 输入 网 址 : http://dev.mysql.com/downloads/ 
installer/， 单 击 转 到 按钮 ， 打 开 MySQL Community Server 5.7.19 下 载 页 面 ， 选 择 
Microsoft Windows 平台 ， 然 后 根据 读者 的 平台 选择 32 位 或 者 64 位 安装 包 ， 在 这 里 
选择 32 位 ， 单 击 右 侧 的 Download 按钮 开始 下 载 ， 如 图 17-1 所 示 。 
EEJPy 在 弹出 的 页 面 中 单 击 Login 按钮 ， 如 图 17-2 所 示 。 

















图 17-1 MySQL 下 载 页 面 图 17-2 单 击 Login 按钮 


pr 这 里 32 位 的 安装 程序 有 两 个 版 本 ， 分 别 为 mysql-installer-web-community 和 
站 ”mysql-installer-communityl， 其 中 前 者 为 在 线 安装 版 本 ， 后 者 为 离线 安装 版 本 。 


EC 弹出 用 户 登 录 页 面 ， 输 入 用 户 名 和 密码 后 ， 单 击 “登录 ”按钮 ， 如 图 17-3 所 示 。 
ES 弹出 开始 下 载 页 面 ， 单 击 Download Now 按钮 ， 即 可 开始 下 载 ， 如 图 17-4 所 示 。 
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CRacte 
Begin Your Download 
登录 
2 et ol To begin your download, please click the Download Now button below. 
a 读本 ® Download Now» 
get cementy 5.7.19 8 ensl 
| mm | MD5: 2578bfc3c30273cee42d77583b8596b5 
Size: 378.8M 
创建 帐户 Signature 
17-3 用户 登录 页 面 图 17-4 开始 下 载 页 面 


Se 如 果 用 户 没有 用 户 名 和 密码 ， 可 以 单 击 “ 创 建 账户 ”链接 进行 注册 即 可 。 


茄 瞳 1DS 灯 册 满 上 内 1DSAW 一 一 般 世 击 席 潍 需 外 小 前 


2. 安装 MySQL 5.7 
MySQL 下 载 完成 后 ， 找 到 下 载 文件 ， 双 击 进行 安装 。 具 体操 作 步 又 如 下 。 
双击 下 载 的 mysql-installer-community-5.7.10.msi 文件 ， 如 图 17-5 所 示 。 


讲 mysql-installer-community-5.7.19.0.msi 


图 17-5 MySQL 安装 文件 名 称 


EEC) 打开 License Agreement( 用 户 许可 证 协议 ) 窗 口 ， 色 选 TI accept the license terms( 我 
接受 许可 协议 ) 复 选 框 ， 单 击 Next( 下 一 步 ) 按 钮 ， 如 图 17-6 所 示 。 

EECS) 打开 Choosing a Setup Type( 安 装 类 型 选择 ) 窗 口 ， 在 其 中 列 出 了 5 种 安装 类 型 ， 分 
别 是 : Developer Default( 默 认 安装 类 型 )、Server only( 仅 作为 服务 器 )、Client only( 仅 
作为 客户 端 )、Full( 完 全 安装 ) 和 Custom( 自 定义 安装 类 型 )。 这 里 选中 Custom( 自 定义 





MySOL. Inetaler eer MySQL Instaler 

















图 17-6 用 户 许可 证 协议 窗口 图 17-7 安装 类 型 窗口 
EEC 打开 Select Products and Features( 产 品 定 制 选择 ) 窗 口 ， 选 择 MySQL Server 
5.7.10-x86 后 ， 单 击 添加 按钮 只 ， 即 可 选择 安装 MySQL 服务 器 。 采 用 同样 的 方 
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法 ， 添 加 Samples and Examples 5.7.10-x86 和 MySQL Documentation 5.7.10-x86 选 
项 ， 如 图 17-8 所 示 。 

EE 单 击 Next( 下 一 步 ) 按 钮 ， 进 入 安装 确认 对 话 框 ， 单 击 Execute( 执 行 ) 按 钮 ， 如 
图 17-9 所 示 。 



































图 17-8 自 定 义 安装 组 件 窗口 图 17-9 准备 安装 对 话 框 


开始 安装 MySQL 文件 ， 安 装 完成 后 在 Status( 状 态 ) 列 表 下 将 显示 Complete( 安 装 
完成 )， 如 图 17-10 所 示 。 





5 

















图 17-10 ”安装 完成 窗口 
17.1.2 配置 MySQL 5.7 


MySQL 安装 完毕 之 后 ， 需 要 对 服务 器 进行 配置 。 具 体 的 配置 步骤 如 下 。 

EEC 在 上 一 节 的 最 后 一 步 中 ， 单 击 Next( 下 一 步 ) 按 钮 ， 进 入 服务 器 配置 窗口 ， 如 图 
17-11 所 示 。 单 击 Next( 下 一 步 ) 按 钮 。 

ECERD9 进入 MySQL 服务 器 配置 窗口 ， 采 用 默认 设置 ， 单 击 Next( 下 一 步 ) 按 钮 ， 如 
图 17-12 所 示 。 
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17-11 ”服务 器 配置 窗口 17-12 MySQL 服务 器 配置 窗口 


MySQL 服务 器 配置 窗口 中 各 个 参数 的 含义 如 下 。 
Server Configuration Type: 该 选项 用 于 设置 服务 器 的 类 型 。 单 击 该 选项 右 侧 的 向 下 按 
钮 ， 即 可 看 到 包括 3 个 选项 ， 如 图 17-13 所 示 。 


Server Configuration Type 
hoos 








| 
ns forth| 
OD Show ANTanced Op 





17-13 ”MySQL 服务 器 的 类 型 


图 17-13 中 3 个 选项 的 具体 含义 如 下 。 
(1) Development Machine( 开 发 机 器 )。 该 选项 代表 典型 个 人 用 桌面 工作 站 。 假 定 机 器 上 运 
行 着 多 个 桌面 应 用 程序 。 将 MySQL 服务 器 配置 成 使 用 最 少 的 系统 资源 。 
(2) Server Machine( 服 务 器 )。 该 选项 代表 服务 器 ，MySQL 服务 器 可 以 同 其 他 应 用 程序 一 
起 运行 ， 如 FTP、Email 和 Web 服务 器 。MySQL 服务 器 配置 成 使 用 适当 比例 的 系统 资源 。 
(3) Dedicated Machine( 专 用 服务 器 )。 该 选项 代表 只 运行 MySQL 服务 的 服务 器 。 假 定 没有 
运行 其 他 服务 程序 ，MySQL 服务 器 配置 成 使 用 所 有 可 用 系统 资源 。 
Sn 作为 初学 者 ， 建 议 选择 Development Machine( 开 发 者 机 器 ) 选 项 ， 这 样 占用 系统 的 
SS 资源 比较 少 。 
EB 打开 设置 服务 器 的 密码 窗口 ， 重 复 输 入 两 次 同样 的 登录 密码 后 ， 单 击 Next( 下 一 
步 ) 按 钮 ， 如 图 17-14 所 示 。 
EEC 打开 设置 服务 器 名 称 窗口 ， 本 案例 设置 服务 器 名 称 为 MySQL， 单 击 Next( 下 一 
步 ) 按 钮 ， 如 图 17-15 所 示 。 
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图 17-14 设置 服务 器 的 登录 密码 17-15 设置 服务 器 的 名 称 


Sn 系统 默认 的 用 户 名 称 为 root， 如 果 想 添加 新 用 户 ， 可 以 单 击 add User( 添 加 用 户 ) 
SS 按钮 进行 添加 。 
ECGD 打开 确认 设置 服务 器 窗口 ， 单 击 Execute( 执 行 ) 按 钮 ， 如 图 17-16 所 示 。 
EDED》 系统 自动 配置 MySQL 服务 器 。 配 置 完成 后 ， 单 击 Finish( 完 成 ) 按 钮 。 即 可 完成 
服务 器 的 配置 ， 如 图 17-17 所 示 。 











图 17-16 确认 设置 服务 器 图 17-17 ”完成 设置 服务 器 


按 Ctrl+AlttDel 组 合 键 ， 打 开 “ 任 务 管理 器 ”对 话 框 ， 可 以 看 到 MySQL 服务 进 
程 mysqld.exe 已 经 启动 了 ， 如 图 17-18 所 示 。 


让 








17-18 ”任务 管理 器 窗口 


至 此 ， 就 完成 了 在 Windows 10 操作 系统 环境 下 安装 MySQL 的 操作 。 


17.2 ”启动 服务 并 登录 MySQL 数据 库 


用 户 可 以 下 载 MySQL 并 安装 。 安 装 完 毕 之 后 ， 需 要 启动 服务 器 进程 ， 不 然 客 户 端 无 法 
连接 数据 库 。 客 户 端 通过 命令 行 工具 登录 数据 库 。 下 面 介 绍 如 何 启动 MySQL 服务 器 和 登录 
MySQL 的 方法 。 


17.2.1 启动 MySQL 服务 


在 前 面 的 配置 过 程 中 ， 已 经 将 MySQL 安装 为 Windows 服务 。 当 Windows 启动 、 停 止 
时 ，MySQL 也 自动 启动 、 人 停止。 不过， 用 户 还 可 以 使 用 图 形 服务 工具 来 控制 MySQL 服务 器 
或 从 命令 行使 用 NET 命令 。 
可 以 通过 Windows 的 服务 管理 器 查看 ， 有 具体 操作 步骤 如 下 。 
EEC 右 击 “开始 ”按钮 ， 在 弹出 的 快捷 菜单 中 选择 “运行 ”菜单 命令 ， 打 开 “ 运 
行 ”对 话 框 ， 输 入 services.msc， 按 Enter 键 确 认 ， 如 图 17-19 所 示 。 
EEPD5 打开 Windows 的 服务 管理 器 ， 在 其 中 可 以 看 到 服务 名 为 MySQL 的 服务 项 ， 其 
右边 状态 为 “正在 运行 ”， 表 明 该 服务 已 经 启动 ， 如 图 17-20 所 示 。 
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Ce we Le 2 





17-19 “运行 ”对 话 框 17-20 ”服务 管理 器 窗口 


由 于 设置 了 MySQL 为 自动 启动 ， 在 这 里 可 以 看 到 ， 服 务 已 经 启动 ， 而 且 启 动 类 型 为 自 
动 。 如 果 没有 “正在 运行 ”字样 ， 说 明 MySQL 服务 未 启动 。 启 动 方法 为 : 打开 “运行 ”对 
话 框 ， 输 入 cmd， 按 Enter 键 确认 。 弹 出 命令 提示 符 界面 。 然 后 输入 net start MySQL， 按 
Enter 键 ， 就 能 启动 MySQL 服务 了 ， 而 停止 MySQL 服务 的 命令 为 net stop MySQL， 如 图 17-21 

也 可 以 直接 双击 MySQL 服务 ， 打开“MySQL 的 属性 ”对 话 框 ， 在 其 中 通过 单 击 “ 启 
动 ” 或 “停止 ”按钮 来 更 改 服务 状态 ， 如 图 17-22 所 示 。 
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丽 管理 中 : Cwindows\system32\cmd.exe - OO x fe 





Bt IEEEF 


SM se 








[er iE 
图 17-21 在 命令 行 中 启动 和 停止 MySQL 17-22 “MySQL 的 属性 ”对 话 框 
Po 输入 的 MySQL 是 服务 的 名 字 。 如 果 读 者 的 MySQL 服务 的 名 字 是 DB 或 其 他 名 


、 字 ， 应 该 输入 net start DB 或 其 他 名 称 。 


17.2.2 ”登录 MySQL 数据 库 

当 MySQL 服务 启动 完成 后 ， 便 可 以 通过 客户 端 来 登录 MySQL 数据 库 。 在 Windows 操 
作 系 统 下 ， 可 以 通过 以 下 两 种 方式 登录 MySQL 数据 库 。 

1. 以 Windows 命令 行 方式 登录 


具体 操作 步骤 如 下 。 
打开 DOS 窗口 ， 输 入 以 下 命令 并 按 Enter 键 确认 ， 如 图 17-23 所 示 。 


cd C:\Program Files\MySQL\MySQL Server 5.7\bin\ 


indows\system3Aemd.exe 





图 17-23 DOS 窗口 


在 DOS 窗口 中 可 以 通过 登录 命令 连接 到 MySQL 数据 库 ， 连 接 MySQL 的 命令 
格式 如 下 : 


mysql -h hostname -ua username -p 


其 中 MySQL 为 登录 命令 ，-h 后 面 的 参数 是 服务 器 的 主机 地 址 ， 在 这 里 客户 端 和 服务 器 
在 同一 台 机 器 上 ， 所 以 输入 localhost 或 者 P 地 址 127.0.0.1; -u 后 面 跟 登 录 数 据 库 的 用 户 名 
称 ， 在 这 里 为 root; -p 后 面 是 用 户 登录 密码 。 

接 下 来 ， 输 入 如 下 命令 : 

mysql -h localhost -u root -p 

按 Enter 键 ， 系 统 会 提示 输入 密码 “Enter password”， 这 里 输入 在 前 面 配置 向 导 中 自己 
设置 的 密码 ， 验 证 正确 后 ， 即 可 登录 到 MySQL 数据 库 ， 如 图 17-24 所 示 。 














说 车 笛 噶 炎 册 外 小 
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图 17-24 Windows 命令 行 登录 窗口 


rp 当 窗口 中 出 现 如 图 17-24 所 示 的 说 明 信 息 ， 命 令 提示 符 变 为 “MySQL>” 时 ， 表 
AS 明 已 经 成 功 登 录 MySQL 服务 器 了 ， 可 以 开始 对 数据 库 进行 操作 。 
2. 使 用 MySQL Command Line Client 登录 


依次 选择 “开始 ”|“ 所 有 程序 ”|MySQLIMySQL Server 5.7IMySQL 5.7 Command Line 
Client 菜单 命令 ， 进 入 密码 输入 窗口 ， 如 图 17-25 所 示 。 


团 MySQL 57 Command Line Client - Unicode 一 口 X 





图 17-25 ”MySQL 命令 行 登录 窗口 
输入 正确 的 密码 之 后 ， 就 可 以 登录 到 MySQL 数据 库 了 。 
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17.3 MySQL 数据 库 的 基本 操作 
下 面 详细 介绍 数据 库 的 基本 操作 。 


17.3.1 创建 数据 库 


创建 数据 库 是 在 系统 磁盘 上 划分 一 块 区 域 用 于 数据 的 存储 和 管理 。 如 果 管 理 员 在 设置 权 
限时 为 用 户 创建 了 数据 库 ， 则 可 以 直接 使 用 ， 否 则 ， 需 要 自己 创建 数据 库 。 在 MySQL 中 创 
建 数据 库 的 基本 SQL 语法 格式 如 下 : 


CREATE DATABASE database name; 


database_name 为 要 创建 的 数据 库 的 名 称 ， 该 名 称 不 能 与 已 经 存在 的 数据 库 重 名 。 
【 例 17.1】 创 建 测试 数据 库 test， 输 入 语句 如 下 : 


CREATE DATABASE test; 


17.3.2 ”查看 数据 库 


数据 库 创建 好 之 后 ， 可 以 使 用 SHOW CREATE DATABASE 声明 查看 数据 库 的 定义 。 
【 例 17.2】 查 看 创建 好 的 数据 库 test 的 定义 ， 输 入 语句 如 下 : 


mysql> SHOW CREATE DATABASE test\G 
训 赤 灾 关 闪闪 碳 册 闪闪 商 奖 闪闪 碳 交 闪闪 闪闪 碳 闪 六 商 庙 关 二 。 工 口 陪 太 六 奖 闪 闪闪 次 奖 关 内 次 奖 类 闪闪 碳 闪 贡 交 次 交 闪闪 六 页 闪闪 
Database: test 
Create Database: CREATE DATABASE ‘test’ /*!40100 DEFAULT CHARACTER SET utf8 
这 
1 row in set (0.00 sec) 


可 以 看 到 ， 如 果 数 据 库 创 建成 功 ， 将 显示 数据 库 的 创建 信息 。 
再 使 用 SHOW DATABASES 语句 来 查看 当前 所 有 存在 的 数据 库 ， 输 入 语句 如 下 : 


mysql> SHOW databases; 


| information_ schema | 
| mysql 1 

| performance_schema | 
| sakila 1 
| 
| 
1 


7 rows in set (0.05 sec) 


可 以 看 到 ， 列 表 中 包含 了 刚 创建 的 数据 库 test 和 其 他 已 经 存在 的 数据 库 的 名 称 。 


17.3.3 ”删除 数据 库 


删除 数据 库 是 将 已 经 存在 的 数据 库 从 磁盘 空间 上 清除 ， 清 除 之 后 ， 数 据 库 中 的 所 有 数据 
也 将 一 同 被 删除 。 删 除数 据 库 语句 和 创建 数据 库 的 命令 相似 。MySQL 中 删除 数据 库 的 基本 语 
法 格式 如 下 : 


DROP DATABASE database name; 


database_name 为 要 删除 的 数据 库 的 名 称 ， 如 果 指 定 的 数据 库 不 存在 ， 则 删除 出 错 。 
【 例 17.3】 删 除 测试 数据 库 test， 输 入 语句 如 下 : 


DROP DATABASE test; 


语句 执行 完毕 之 后 ， 数 据 库 test 将 被 删除 。 再 使 用 SHOW CREATE DATABASE 声明 查 
看 数据 库 的 定义 ， 结 果 如 下 : 


mysql> SHOW CREATE DATABASE test\G 

ERROR 1049 (42000): Unknown database 'test"' 
ERROR: 

No query specified 


执行 结果 给 出 一 条 错误 信息 ERROR 1049 (42000): Unknown database “test"， 即 数据 库 
test 已 不 存在 ， 删 除 成 功 。 
en 使 用 DROP DATABASE 命令 时 要 非常 谨慎 。 在 执行 该 命令 时 ，MySQL 不 会 给 出 
汽 ”任何 提醒 确认 信息 。 DROP DATABASE 声明 删除 数据 库 后 ， 数 据 库 中 存储 的 所 有 数 
据 表 和 数据 也 将 一 同 被 删除 ， 而 且 不 能 恢复 。 
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17.3.4 选择 数据 库 

用 户 创建 了 数据 库 后 ， 并 不 能 使 用 SQL 语句 操作 该 数据 库 ， 还 需要 使 用 USE 语句 选择 
该 数据 库 。 有 具体 的 语法 如 下 : 

USE 数据 库 名 ; 

【 例 17.4】 选 择 数 据 库 test， 输 入 语句 如 下 : 


mysql> USE test; 
Database changed 


17.4 ”MySQL 数据 表 的 基本 操作 


下 面 详细 介绍 数据 表 的 基本 操作 ， 主 要 内 容 包括 : 创建 数据 表 、 查 看 数据 表 结 构 、 修 改 
数据 表 、 删 除数 据 表 。 
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17.4.1 创建 数据 表 


数据 表 属 于 数据 库 。 在 创建 数据 表 之 前 ， 应 该 使 用 语句 “USE < 数据 库 名 >” 指 定 操作 是 
在 哪个 数据 库 中 进行 的 。 如 果 没 有 选择 数据 库 ， 会 抛 出 No database selected 错误 。 

创建 数据 表 的 语句 为 CREATE TABLE， 语 法 格式 如 下 : 

CREATE TABLE < 表 名 > 

( 


字段 名 1， 数 据 类 型 [ 列 级 别 约束 条 件 ] [默认 值 ] ， 
字段 名 2， 数 据 类 型 [ 列 级 别 约束 条 件 ] [默认 值 ] ， 


[ 表 级 别 约束 条 件 ] 





) 7 

使 用 CREATE TABLE 创建 表 时 ， 必 须 指定 以 下 信息 。 

(1) 要 创建 的 表 的 名 称 ， 不 区 分 大 小 写 ， 不 能 使 用 SQL 语言 中 的 关键 字 ， 如 DROP、 
ALTER、INSERT 等 。 

(2) 数据 表 中 每 一 个 列 (字段 ) 的 名 称 和 数据 类 型 ， 如 果 创 建 多 个 列 ， 要 用 逗号 隔 开 。 

【 例 17.5】 创 建 员工 表 tb_emp1， 结 构 如 表 17-1 所 示 。 


表 17-1 tb_emp1 表 的 结构 


字段 名 称 数据 类 型 





ia 
vame 
depud 
salary 


首先 创建 数据 库 ，SQL 语句 如 下 : 


CREATE DATABASE test; 


选择 创建 表 的 数据 库 ，SQL 语句 如 下 : 


USE test; 
创建 tb_empl 表 ，SQL 语句 如 下 : 


CREATE TABLE tb_empl 
( 

id INT(11), 

name VARCHAR(25), 
deptId INT(11), 
salary FLOAT 

Fe 


语句 执行 后 ， 便 创建 了 一 个 名 称 为 tb_emp1l 的 数据 表 ， 使 用 SHOW TABLES 语句 查看 数 
据 表 是 否 创建 成 功 ，SQL 语句 如 下 : 


mysql> SHOW TABLES; 
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二 -一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 + 
| Tables in_ test | 

4 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 + 

| tb_empl | 

二 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 一 十 


1 row in set (0.00 sec) 


可 以 看 到 ，test 数据 库 中 已 经 有 了 数据 表 tb_emp1， 数 据 表 创 建成 功 。 


17.4.2 ”查看 数据 表 的 结构 


使 用 SQL 语句 创建 好 数据 表 之 后 ， 可 以 查看 表 结 构 的 定义 ， 以 确认 表 的 定义 是 否 正确 。 
在 MySQL 中 ， 查 看 表 结 构 可 以 使 用 DESCRIBE 和 SHOW CREATE TABLE 语句 。 这 里 将 针 
对 这 两 个 语句 分 别 进行 详细 的 讲解 。 

DESCRIBE/DESC 语句 可 以 查看 表 的 字段 信息 ， 其 中 包括 字段 名 、 字 段 数据 类 型 、 是 否 
为 主键 、 是 否 有 默认 值 等 。 语 法 格式 如 下 : 

DESCRIBE 表 名 ; 

或 者 简写 为 : 

DESC 表 名 ; 

【 例 17.6】 使 用 DESCRIBE 查看 表 tb_emp1l 的 表 结构 。 

查看 tb_empl 表 结 构 ，SQL 语句 如 下 : 


mysql> DESC tb_empl; 
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+ 一 -一 一 一 一 + 一 一 一 一 一 一 一 一 一 一 一 一 一 + 一- 一 -一 + 一 一 一 一 一 +- 一 一 一 一 一 一 一 一 + 一 一 一 一 一 一 + 

| Field | Type | Null | Key | Default | Extra | 

+ 一 -一 一 -一 +------------- 一 +----- 一 二 一- 一 -一 +-------- 一 +- 一 -一 一 一 一 十 
De! | 1 YES | | NULL | | 

| name | varchar(25) | YES | | NULL | 1 

| deptId | int (11) 1 YES | | NULL | | 

| salary | float 1 YES | | NULL | 1 

+- 一 -一 一 一 一 一 一 二 -一 一 一 一 一 一 一 一 一 一 一 一 + 一 一 一 一 一 一 一 + 一 一 一 一 + 一 一 一 一 一 一 一 一 + 一 一 一 一 一 一 + 


其 中 ， 各 个 字段 的 含义 分 别 解释 如 下 。 

@ NULL: 表示 该 列 是 否 可 以 存储 NULL 值 。 

@ Key: 表示 该 列 是 否 已 编制 索引 。PRI 表示 该 列 是 表 主 键 的 一 部 分 UNI 表示 该 列 是 

UNIQUE 索引 的 一 部 分 ，MUL 表示 在 列 中 某 个 给 定 值 允许 出 现 多 次 。 

@ ”Default 表示 该 列 是否 有 默认 值 ， 如 果 有 的 话 值 是 多 少 。 

@ Extra: 表示 可 以 获取 的 与 给 定 列 有 关 的 附加 信息 ， 如 AUTO_INCREMENT 等 。 

SHOW CREATE TABLE 语句 可 以 用 来 显示 创建 表 时 的 CREATE TABLE 语句 ， 其 语法 
格式 如 下 : 


SHOW CREATE TABLE < 表 名 \G>; 
Sn 使 用 SHOW CREATE TABLE 语句 ， 不 仅 可 以 查看 表 创 建 时 的 详细 语句 ， 而 且 还 
外 可 以 查看 存储 引擎 和 字符 编码 。 
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如 果 不 加 “\G” 参 数 ， 显 示 的 结果 可 能 非常 混乱 ， 加 上 参数 “\G” 之 后 ， 可 使 显示 结果 
更 加 直观 ， 易 于 查看 。 
【 例 17.7】 使 用 SHOW CREATE TABLE 查看 表 tb_empl 的 详细 信息 ，SQL 语句 如 下 : 


mysql> SHOW CREATE TABLE tb_empl; 





| Table 








| fruits | CREATE TABLE ‘fruits. ( 

fid. char(10) NOT NULL, 

“sid. int(11) NOT NULL, 

“f name. char(255) NOT NULL, 

‘fprice decimal(8,2) NOT NULL, 

PRIMARY KEY (‘f id), 

KEY ‘index name. (‘f name ), 

KEY "index:id price (fid ; foprice ) 
) ENGINE=InnoDB DEFAULT CHARSET=gb2312 | 
十 一 十 一 








使 用 参数 “\G” 之 后 的 结果 如 下 : 


mysql> SHOW CREATE TABLE tb_empl\G 
灵 赤 广大 二 潍 赤 姑姑 尖 去 雪 关 二 潍 友 大 赤 二 到 友 坟 大坪 二 二 天 了] 。 工 口 于 太 六 身 兴 闪闪 奖 商 闪闪 商 交 闪闪 闪闪 闪闪 闪 商 交 庙 认 太 页 闪 
Table: tb_empl 
Create Table: CREATE TABLE ‘tb empl~ ( 
“id. int(11) DEFAULT NULL, 
‘name’ Varchar (25) DEFAULT NULL, 
“deptId ”int(11) DEFAULT NULL, 
“salary. float DEFAULT NULL 
) ENGINE=InnoDB DEFAULT CHARSET=gb2312 
1 row in set (0.00 sec) 


17.4.3 ”修改 数据 表 结 构 
MySQL 是 通过 ALTER TABLE 语句 来 修改 表 结 构 的 ， 具 体 语法 格式 如 下 : 


ALTER[IGNORE] TABLE 数据 表 名 alter_spec[, alter spec]... 
其 中 alter_spec 定义 要 修改 的 内 容 ， 语 法 格式 如 下 : 


ADD [COLUMN] create definition [FIRST|IAFTER column_name]  // 添 加 新 字段 


ALTER [COLUMN] col name{SET DEFAULT literal |DROP DEFAULT}// 修 改 字段 名 称 
CHANGE [COLUMN] old col name create definition // 修 改 字段 类 型 


| ADD INDEX [index_name] (index col name,...) // 添 加 索引 名 称 
| ADD PRIMARY KEY (index col name ---) // 添 加 主键 名 称 
| ADD UNIQUE [index name] (index col name,...) // 添 加 唯一 索引 
1 
| 


| MODIFY [COLUMN] create definition // 添 加 子 句 定义 类 型 
| DROP [COLUMN] col name // 删 除 字段 名 称 

| DROP PRIMARY KEY // 删 除 主键 名 称 

| DROP INDEX idex name // 删 除 索引 名 称 

| RENAME [RS] new tbl] name // 更 改 表 名 

1 


table_options 


【 例 17.8】 将 数据 表 tb_deptl 中 name 字段 的 类 型 由 VARCHAR(22) 改 成 VARCHAR(30)。 
输入 如 下 SQL 语句 并 执行 : 


ALTER TABLE tb deptl1 MODIFY name VARCHAR(30); 


17.4.4 ”删除 数据 表 


删除 数据 表 就 是 将 数据 库 中 已 经 存在 的 表 从 数据 库 中 删除 。 注 意 ， 在 删除 表 的 同时 ， 表 
的 定义 和 表 中 所 有 的 数据 均 会 被 删除 。 因 此 ， 在 进行 删除 操作 前 ， 最 好 对 表 中 的 数据 做 个 备 
份 ， 以 兔 产 生 无 法 挽回 的 后 果 。 

在 MySQL 中 ,使 用 DROP TABLE 可 以 一 次 删除 一 个 或 多 个 没有 被 其 他 表 关 联 的 数据 
表 。 语 法 格式 如 下 : 


DROP TABLE [IF EXISTS] 表 1, 表 2，..., 表 n; 


其 中 ，“ 表 n” 指 要 删除 的 表 的 名 称 ， 后 面 可 以 同时 删除 多 个 表 ， 只 需 将 要 删除 的 表 名 依 
次 写 在 后 面 ， 相 互 之 间 用 逗号 隔 开 即 可 。 如 果 要 删除 的 数据 表 不 存在 ， 则 MySQL 会 提示 一 
条 错误 信息 “ERROR 1051 (42S02): Unknown table ' 表 名 '”。 参 数 IF EXISTS 用 于 在 删除 前 判 
断 删 除 的 表 是 否 存 在 ， 加 上 该 参数 后 ， 再 删除 表 的 时 候 ， 如 果 表 不 存在 ，SQL 语句 可 以 顺利 
执行 ， 但 是 会 发 出 警告 (warning)。 

【 例 17.9】 删 除数 据 表 tb_dept2，SQL 语句 如 下 : 


DROP TABLE IF EXISTS tb_dept2; 
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17.5 ”MySQL 语句 的 操作 
下 面 讲述 MySQL 语句 的 基本 操作 。 


17.5.1 插入 记录 


使 用 基本 的 INSERT 语句 插入 数据 ， 要 求 指定 表 名 称 和 插入 到 新 记录 中 的 值 。 基 本 语法 
格式 如 下 : 


INSERT INTO table name (column list) VALUES (value list); 


table name 指定 要 插入 数据 的 表 名 ; column _list 指定 要 插入 数据 的 那些 列 ，value_list 指 
定 每 个 列 应 对 应 插入 的 数据 。 注 意 ， 使 用 该 语句 时 ， 字 段 列 和 数据 值 的 数量 必须 相同 。 
在 MySQL 中 ， 可 以 一 次 性 插入 多 行 记 录 ， 各 行 记 录 直 接 由 逗号 分 隔 即 可 。 
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【 例 17.10】 创 建 数据 表 tmp7， 定 义 数据 类 型 为 TIMESTAMP 的 字段 ts， 向 表 中 插入 值 
'19950101010101' ， '950505050505' ， '1996-02-02 02:02:02' ，'97@03@03 03@03@03' ， 
121212121212，NOWO，SQL 语句 如 下 : 


CREATE TABLE tmp7 (ts TIMESTAMP); 
INSERT INTO tmp7(ts) values('19950101010101°'), 


('950505050505"'), 
("1996-02-02 02:02:02"'), 
('97@03@03 03@038@03'), 
(121212121212), 

(NOW()); 


17.5.2 ”查询 记录 
MySQL 从 数据 表 中 查询 数据 的 基本 语句 为 SELECT 语句 。SELECT 语句 的 基本 格式 如 下 : 


SELECT 
{* | < 字段 列表 >} 


E 

FROM < 表 1>,< 表 2>,... 

[WHERE < 表达 式 > 

[GROUP BY <group by definition>] 

[HAVING <expression> [{<operator> <expression>}...]] 
[ORDER BY <order by definition>] 

[LIMIT [<offset>,] <row count>] 


] 


SELECT [字段 1, 字 段 2, . . .字段 n] 
FROM [ 表 或 视图 ] 
WHERE [查询 条 件 ] ; 


其 中 ， 各 条 子 句 的 含义 如 下 。 


{* | < 字段 列表 >}: 包含 星 号 通配符 和 字段 列表 ， 表 示 查 询 的 字段 ， 其 中 字段 列 至 少 
包含 一 个 字段 名 称 ， 如 果 要 查询 多 个 字段 ， 多 个 字段 之 间 用 逗号 隔 开 ， 最 后 一 个 字 
段 后 不 要 加 逗号 。 

FROM < 表 1>,< 表 2>，...: 表 1 和 表 2 表示 查询 数据 的 来 源 ， 可 以 是 单个 或 者 多 个 。 
WHERE: 该 子 句 是 可 选项 ， 如 果 选 择 该 项 ， 将 限定 查询 行 必须 满足 的 查询 条 件 。 
GROUP BY < 字段 >: 该 子 句 告诉 MySQL 如 何 显示 查询 出 来 的 数据 ， 并 按照 指定 的 
字段 分 组 。 

ORDER BY < 字段 >: 该 子 句 告诉 MySQL 按 什么 样 的 顺序 显示 查询 出 来 的 数据 ， 可 
以 进行 的 排序 有 升序 (ASC)、 降 序 (DESC)。 

[LIMIT [<offset>,] <row count>]: 该 子 句 指明 每 次 显示 查询 出 来 的 数据 条 数 。 


【 例 17.11】 从 fruits 表 中 获取 f name 和 fprice 两 列 ，SQL 语句 如 下 : 


SELECT f name, f price FROM fruits; 


17.5.3 ”修改 记录 


表 中 有 数据 之 后 ， 接 下 来 可 以 对 数据 进行 更 新 操作 。 在 MySQL 中 使 用 UPDATE 语句 更 
新 表 中 的 记录 ， 可 以 更 新 特定 的 行 或 者 同时 更 新 所 有 的 行 。 基 本 语法 格式 如 下 : 


UPDRTE table_name 
SET column_namel = valuel,column name2=value2,...,column namen=valuen 
WHERE (condition); 


column namel,column name?2,...,column namen 为 指定 更 新 的 字段 的 名 称 ; valuel， 
value2,…,valuen 为 相对 应 的 指定 字段 的 更 新 值 ，condition 指定 更 新 的 记录 需要 满足 的 条 件 。 
更 新 多 个 列 时 ， 每 个 “ 列 - 值 ” 对 之 间 用 去 号 隔 开 ， 最 后 一 列 之 后 不 需要 逗号 。 

【 例 17.12】 在 person 表 中 ， 更 新 id 值 为 11 的 记录 ， 将 age 字段 值 改 为 1 5， 将 name 字 
段 值 改 为 LiMing，SQL 语句 如 下 : 


UPDATE person SET age = 15, name='LiMing' WHERE id = 117 


17.5.4 删除 记录 


从 数据 表 中 删除 数据 使 用 DELETE 语句 。DELETE 语句 允许 WHERE 子 句 指定 删除 条 
件 。DELETE 语句 的 基本 语法 格式 如 下 : 

DELETE FROM table name [WHERE <condition>]; 

table name 指定 要 执行 删除 操作 的 表 ; [WHERE <condition>] 为 可 选 参数 ， 指 定 删 除 条 
件 ; 如果 没有 WHERE 子 句 ，DELETE 语句 将 删除 表 中 的 所 有 记录 。 

【 例 17.13】 在 person 表 中 ， 删 除 id 等 于 11 的 记录 ，SQL 语句 如 下 : 


mysql> DELETE FROM person WHERE id = 11; 
Query OK, 1 row affected (0.02 sec) 


17.6 MySQL 数据 库 的 备份 与 还 原 


MySQL 提供 了 多 种 方法 对 数据 进行 备份 和 还 原 。 下 面 介绍 数据 备份 和 数据 还 原 的 相关 
知识 。 


17.6.1 数据 备份 


数据 备份 是 数据 库 管 理 员 非 常 重要 的 工作 。 系 统 意外 崩溃 或 者 硬件 的 损坏 都 可 能 导致 数 
据 库 的 丢失 。 因 此 ，MySQL 管理 员 应 该 定期 地 备份 数据 库 ， 使 得 在 意外 情况 发 生 时 ， 尽 可 能 
减少 损失 。 下 面 介绍 数据 备份 的 3 种 方法 。 
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1. 使 用 mysqldump 命令 备份 


mysqldump 是 MySQL 提供 的 一 个 非常 有 用 的 数据 库 备份 工具 。mysqldump 命令 执行 时 ， 
可 以 将 数据 库 备份 成 一 个 文本 文件 。 该 文件 中 实际 上 包含 了 多 个 CREATE 和 INSERT 语句 ， 
使 用 这 些 语句 可 以 重新 创建 表 和 插入 数据 。 

mysqldump 备份 数据 库 语 句 的 基本 语法 格式 如 下 : 


mysqldump -u user -h host -ppassword dbname[tbname, [tbname...]]> 
filename.sql 


其 中 ，user 表示 用 户 名 称 ; host 表示 登录 用 户 的 主机 名 称 ; password 为 登录 密码 ; 
dbname 为 需要 备份 的 数据 库 名 称 ; tbname 为 dbname 数据 库 中 需要 备份 的 数据 表 ， 可 以 指定 
多 个 需要 备份 的 表 ; 右 箭头 符号 “>” 告 诉 mysqldump 将 备份 数据 表 的 定义 和 数据 写 入 备份 
文件 ，filename.sql 为 备份 文件 的 名 称 。 

【 例 17.14】 使 用 mysqldump 命令 备份 数据 库 中 的 所 有 表 ， 执 行 过 程 如 下 。 

为 了 更 好 地 理解 mysqldump 工具 如 何 工 作 ， 这 里 给 出 一 个 完整 的 数据 库 例 子 。 首 先 登 录 
MySQL， 按 下 面 的 数据 库 结构 创建 booksDB 数据 库 和 各 个 表 ， 并 插入 数据 记录 。 数 据 库 和 表 
定义 如 下 : 

CREATE DATABASE booksDB; 

use booksDB; 





CREATE TABLE books 
( 
bk_id INT NOT NULL PRIMARY KEY, 
bk title VARCHAR(50) NOT NULL, 
copyright YEAR NOT NULL 
); 
INSERT INTO books 
VALUES (11078, 'Learning MySQL', 2010), 
(11033, 'Study Html', 2011), 
(11035, 'How to use php', 2003), 
(11072, 'Teach yourself javascript', 2005), 
(11028, 'Learning C++', 2005), 
(11069, 'MySQL professional', 2009), 
(11026, 'Guide to MYSQL 5.7', 2008), 
(11041, "Inside VC++', 2011); 


CREATE TABLE authors 
( 
auth_id INT NOT NULL PRIMARY KEY, 
auth name VARCHAR(20), 
auth_gender CHAR(1) 
); 
INSERT INTO authors 
VALUES (1001， 'Writerx’' ,'f'), 
(10027 "WriterA’ wf£")r 
(1003, "WriterB’' , 'm’'), 
(1004, 'WriterC’ ,f°'), 
(1011, 'WriterD® ,f°), 
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(1012, 'WriterE' ,'m'), 
(1013, 'WriterF' ,'m'), 
(LO0L4y "WriterG rE)y 
(UOLSy WriterH Ey 


CREATE TABLE authorbook 
( 
auth_id INT NOT NULL, 
bk_id INT NOT NULL, 
PRIMARY KEY (auth id, bk id), 
FOREIGN KEY (auth id) REFERENCES authors (auth id), 
FOREIGN KEY (bk_id) REFERENCES books (bk_id) 


INSERT INTO authorbook 
VALUES (1001, 11033), (1002, 11035), (1003, 11072), (1004, 11028), 
(1011, 11078), (1012, 11026), (1012, 11041), (1014, 11069); 


完成 数据 插入 后 ， 打 开 操 作 系 统 命 令 行 输入 窗口 ， 输 入 备份 命令 如 下 : 


C:\> mysqldump -u root -p booksdb > C:/backup/booksdb 20180301.sql 
Enter password: ** 


输入 密码 之 后 ，MySQL 便 对 数据 库 进行 了 备份 ， 在 C:\backup 文件 夹 下 面 查 看 刚才 备份 
过 的 文件 ， 使 用 文本 查看 器 打开 文件 ， 可 以 看 到 其 部 分 文件 内 容 大 致 如 下 : 


-- MySQL dump 10.13 Distrib 5.7.19, for Win32 (x86) 


-- Host: localhost Database: booksDB 
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-- Server version 5.7.19 


/*!140101 SET @OLD_ CHARACTER_SET CLIENT=@@CHARACTER_SET_CLIENT */; 
/*!140101 SET @OLD CHARACTER_SET_ RESULTS=@@CHARACTER_SET RESULTS */; 
/*!140101 SET @OLD_COLLATION_CONNECTION=@@COLLATION_CONNECTION */; 
/*!140101 SET NAMES utf8 */; 

/*!140103 SET QOLD TIME ZONE=@@TIME ZONE */; 

/*140103 SET TIME ZONE='+00:00" */; 

/*!140014 SET @OLD UNIQUE CHECKS=@Q@UNIQUE CHECKS, UNIQUE CHECKS=0 */; 
/*!140014 SET @OLD FOREIGN KEY CHECKS=@@FOREIGN_KEY_ CHECKS, FOREIGN_KEY_ 
CHECKS=0 */; 

/*!140101 SET Q@OLD SQL MODE=Q@@SQL MODE, SQL MODE= 
'NO_AUTO_VALUE_ON_ZERO' */; 

/*!140111 SET Q@OLD_ SQL NOTES=@@SQL NOTES, SQL NOTES=0 */; 


-— Table structure for table "authorbook'" 


DROP TABLE IF EXISTS "authorbook'7 
/*!140101 SET @saved cs client = @@character set client */7 
/*140101 SET character set client = utf8 */; 
CREATE TABLE 'authorbook' ( 
"auth id' int(11) NOT NULL, 
'bk_id' int(11) NOT NULL, 
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PRIMARY KEY ('auth id', 'bk id'), 
KE "bedid, ("Dieidu dy 
CONSTRAINT ‘authorbook ibfk 1' FOREIGN KEY ('auth id') 
REFERENCES "authors' ('auth id'), 
CONSTRAINT ‘authorbook ibfk 2' FOREIGN KEY ('bk id') 
REFERENCES "books' ('bk id') 
) ENGINE=InnoDB DEFAULT CHARSET=utf8; 
/*!140101 SET character set client = @saved cs client */7 


LOCK TABLES "authorbook' WRITE; 

/*!140000 ALTER TABLE ‘'authorbook' DISABLE KEYS */; 

INSERT INTO "authorbook' VALUES (1012,11026), (1004,11028), (1001,11033), 
(1002,11035), (1012, 11041), (1014,11069), (1003,11072), (1011,11078); 
/*!140000 ALTER TABLE 'authorbook' ENABLE KEYS */; 

UNLOCK TABLES; 


. .省略 部 分 内 容 

/x 了。 40103 SET TIME ZONE=@OLD_TIME ZONE */; 

/*!140101 SET SQL MODE=@OLD SQL MODE */; 

/*!40014 SET FOREIGN_KEY_ CHECKS=@OLD_ FOREIGN_ KEY _ CHECKS */; 

/*!140014 SET UNIQUE CHECKS=@OLD_UNIQUE CHECKS */; 

/*!140101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */; 

/*!140101 SET CHARACTER_SET_RESULTS=@OLD_CHARACTER_SET_ RESULTS */; 

/*!140101 SET COLLATION_CONNECTION=@OLD COLLATION_CONNECTION */; 

/*!140111 SET SQL_NOTES=@OLD_ SQL_NOTES */; 

-- Dump completed on 2011-08-18 10:44:08 

可 以 看 到 ， 备 份 文件 包含 了 一 些 信息 ， 文 件 开头 首先 表明 了 备份 文件 使 用 的 mysqldump 
工具 的 版 本 号 ; 然后 是 备份 账户 的 名 称 和 主机 信息 ， 以 及 备份 的 数据 库 的 名 称 ， 最 后 是 
MySQL 服务 器 的 版 本 号 ， 在 这 里 为 5.7.19。 

备份 文件 接 下 来 的 部 分 是 一 些 SET 语句 ， 这 些 语句 将 一 些 系统 变量 值 赋 给 用 户 定义 变 
量 ， 以 确保 被 恢复 的 数据 库 的 系统 变量 与 原来 备份 时 的 变量 相同 。 例 如 : 


/*!140101 SET @OLD_CHARACTER_ SET_CLIENT=@@CHARACTER SET_CLIENT */; 


该 SET 语句 将 当前 系统 变量 character_set_client 的 值 赋 给 用 户 定义 变量 @old_character_ 
set_client。 其 他 变量 与 此 类 似 。 

备份 文件 的 最 后 几 行 MySQL 使 用 SET 语句 恢复 服务 器 系统 变量 原来 的 值 。 例 如 : 

/*!140101 SET CHARACTER SET_CLIENT=@OLD CHARACTER_ SET_CLIENT */7 

该 语句 将 用 户 定 义 的 变量 @old_character_set_client 中 保存 的 值 赋 给 实际 的 系统 变量 
character set_client。 

备份 文件 中 的 “--” 字 符 开头 的 行为 注释 语句 ， 以 “/*1” 开 头 、“*/” 结 尾 的 语句 为 可 执 
行 的 MySQL 注释， 这些 语句 可 以 被 MySQL 执行 ， 但 在 其 他 数据 库 管 理 系统 中 将 被 作为 注释 





忽略 ， 这 可 以 提高 数据 库 的 可 移植 性 。 
另外 注意 到 ， 备 份 文件 开始 的 一 些 语句 以 数字 开头 ， 这 些 数字 代表 了 MySQL 版 本 号 。 
这 些 数字 告诉 我 们 ， 这 些 语句 只 有 在 指定 的 MySQL 版 本 或 者 比 该 版 本 高 的 情况 下 才能 执 
行 。 例 如 ，40101 表明 这 些 语句 只 有 在 MySQL 版 本 号 为 4.01.01 或 更 高 的 条 件 下 才 可 以 被 执行 。 
在 前 面 介 绍 的 mysqldump 语法 中 介绍 过 ，mysqldump 还 可 以 备份 数据 中 的 某 个 表 ， 其 语 
法 格式 如 下 : 


mysqldump -u user -h host -p dbname [tbname, [tbname...]] > filename.sql 


tbname 表示 数据 库 中 的 表 名 ， 多 个 表 名 之 间 用 空格 隔 开 。 
备份 表 和 备份 数据 库 中 所 有 表 的 语句 中 不 同 的 地 方 在 于 ， 要 在 数据 库 名 称 dbname 之 后 指 
定 需要 备份 的 表 名 称 。 
【 例 17.15】 备 份 booksDB 数据 库 中 的 books 表 ， 输 入 语句 如 下 : 


mysqldump -u root -p booksDB books > C:/backup/books_20180301.sql 


该 语句 创建 名 称 为 books_ 20180301.sql 的 备份 文件 ， 文 件 中 包含 了 前 面 介绍 的 SET 语句 
等 内 容 。 不 同 的 是 ， 该 文件 只 包含 books 表 的 CREATE 和 INSERT 语句 。 

如 果 要 使 用 mysqldump 备份 多 个 数据 库 ， 需 要 使 用 --databases 参数 。 备 份 多 个 数据 库 的 
语法 格式 如 下 : 


mysqldump -u user -h host -p --databases [dbname, [dbname...]]> 
filename.sql 


使 用 --databases 参数 之 后 ， 必 须 指定 至 少 一 个 数据 库 的 名 称 ， 多 个 数据 库 名 称 之 间 用 空格 
隔 开 。 
【 例 17.16】 使 用 mysqldump 备份 booksDB 和 test 数据 库 ， 输 入 语句 如 下 : 


mysqldump -u root -p --databases booksDB test > C:\backup\books_ testDB_ 
20180301.sql 


该 语句 创建 名 称 为 books_testDB_20180301.sql 的 备份 文件 ， 文 件 中 包含 了 创建 两 个 数据 
库 booksDB 和 test 所 必需 的 所 有 语句 。 
另外 ， 使 用 --all-databases 参数 可 以 备份 系统 中 所 有 的 数据 库 ， 输 入 语句 如 下 : 


mysqldump -u user -h host -p --all-databases > filename.sql 


使 用 参数 --all-databases 参数 时 ， 不 需要 指定 数据 库 名 称 。 
【 例 17.17】 使 用 mysqldump 备份 服务 器 中 的 所 有 数据 库 ， 输 入 语句 如 下 : 


mysqldump -u root -p --all-databases > C:/backup/alldbinMysQL.sql 


该 语句 创建 名 称 为 alldbinMySQL.sql 的 备份 文件 ， 文 件 中 包含 了 对 系统 中 所 有 数据 库 的 
备份 信息 。 


Sn 在 服务 器 上 进行 备份 ， 并 且 表 均 为 MyISAM 表 时 ， 应 考虑 使 用 mysqlhotcopy， 


外。 因为 这 可 以 更 快 地 进行 备份 和 恢复 。 





mysqldump 还 有 一 些 其 他 选项 可 以 用 来 指定 备份 过 程 ， 如 --opt 选项 ， 该 选项 将 打开 
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--quick、--add-locks、--extended-insert 等 多 个 选项 。--opt 选项 可 以 提供 最 快速 的 数据 库 转 储 。 


mysqldump 的 其 他 常用 选项 如 下 。 

®@ ”--add-drop-database: 在 每 个 CREATE DATABASE 语句 前 添加 DROP DATABASE 
语句 。 

®@ ”--add-drop-tables: 在 每 个 CREATE TABLE 语句 前 添加 DROP TABLE 语句 。 

e@  --add-locking: 用 LOCK TABLES 和 UNLOCK TABLES 语句 引用 每 个 表 转 储 。 重 载 
转 储 文件 时 插入 得 更 快 。 

@ 。 --all-database,-A: 转 储 所 有 数据 库 中 的 所 有 表 。 与 使 用 --database 选项 相同 ， 在 命令 
行 中 命名 所 有 数据 库 。 


®@ ”--comments[=0|1]: 如 果 设 置 为 0， 则 禁止 转 储 文件 中 的 其 他 信息 ， 如 程序 版 本 、 服 
务 器 版 本 和 主机 。--skip-comments 与 --comments=0 的 结果 相同 。 默 认 值 为 1， 即 包 
括 额 外 信息 。 

@  --compact: 产生 少量 输出 。 该 选项 禁用 注释 并 启用 --skip-add-drop-tables 、--no-set- 
names、--skip-disable-keys 和 --skip-add-locking 选项 。 

®@ ”--compatible=name: 产生 与 其 他 数据 库 系统 或 旧 的 MySQL 服务 器 更 兼容 的 输出 。 值 
可 以 为 ansi、mysql323 、mysql40 、postgresql 、 oracle 、mssql 、db2 、maxdb 、 





no_key_options、no_tables_options 或 者 no_field options。 

--complete-insert,-c: 使 用 包括 列 名 的 完整 的 INSERT 语句 。 

---debug[=debug_options],-# [debug_options]: 写 调 试 日 志 。 

--delete,-D: 导入 文本 文件 前 清空 表 。 

--default-character-set=charset: 用 charset 作为 默认 字符 集 。 若 没有 指定 ，mysqldump 

使 用 utf8。 

@  --delete-master-logs: 在 主 复制 服务 器 上 ， 完 成 转 储 操作 后 删除 二 进 制 日 志 。 该 选项 
自动 启用 -master-data。 

@。 --extended-insert,-e: 使 用 包括 几 个 VALUES 列表 的 多 行 INSERT 语法 。 这 样 使 转 储 
文件 更 小 ， 重 载 文件 时 可 以 加 速 插入 。 

®@  --flush-logs,-E: 开始 转 储 前 刷新 MySQL 服务 器 日 志文 件 。 要 求 RELOAD 权限 。 

@  --force,-f: 在 表 转 储 过 程 中 ， 即 使 出 现 SQL 错误 也 继续 。 

e@。 --lock-all-tables,-x: 对 所 有 数据 库 中 的 所 有 表 加 锁 。 在 整体 转 储 过 程 中 通过 全 局 锁定 
来 实现 。 该 选项 自动 关闭 --single-transaction 和 --lock-tables。 

@ ”--lock-tables,-l: 开始 转 储 前 锁定 所 有 表 。 用 READ LOCAL 锁定 表 以 允许 并 行 插入 
MyISAM 表 。 对 于 事务 表 ( 如 InnoDB 和 BDB)，--single-transaction 是 一 个 更 好 的 选 
项 ， 因 为 它 根本 不 需要 锁定 表 。 

@ --no-create-db,.-m: 该 选项 禁用 CREATE DATABASE /*!32312 IF NOT EXISTS*/ 
db_name 语句 ， 如 果 给 出 --database 或 --all-database 选项 ， 则 包含 到 输出 中 。 

@ 。--no-create-info,-t: 只 导出 数据 ， 而 不 添加 CREATE TABLE 语句 。 

e@ ”--no-data,-d: 不 写 表 的 任何 行 信息 ， 只 转 储 表 的 结构 。 
--opt: 该 选项 是 速记 ， 等 同 于 指定 --add-drop-tables--add-locking，--create-option，-- 
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disable-keys--extended-insert，--lock-tables-quick 和 --set-charset。 它 可 以 快速 进行 转 储 
操作 并 产生 一 个 能 很 快 装 入 MySQL 服务 器 的 转 储 文件 。 该 选项 默认 开启 ， 但 可 以 
用 --skip-opt 禁用 。 要 想 禁 用 -opt 启用 的 选项 ， 可 以 使 用 --skip 形式 ， 如 --skip-add- 
drop-tables 或 --skip-quick。 
®@  --password[=password],-p[password]: 当 连 接 服 务 器 时 使 用 的 密码 。 如 果 使 用 短 选项 
形式 (-p)， 选 项 和 密码 之 间 不 能 有 空格 。 如 果 在 命令 行 中 --password 或 -p 选项 后 面 没 
有 密码 值 ， 则 提示 输入 一 个 密码 。 
e@ 。--port=port_ num,-P port num: 用 于 连接 的 TCP/IP 端口 号 。 
®@ ”--protocol={TCP | SOCKET | PIPE | MEMORY}: 使 用 的 连接 协议 。 
® 。 --Teplace,-r: --replace 和 --ignore 选项 控制 蔡 换 或 负责 唯一 键 值 已 有 记录 的 输入 记 
录 的 处 理 。 如 果 指 定 --replace， 新 行 蔡 换 有 相同 的 唯一 键 值 的 已 有 行 ， 如 果 指 定 
--ignore， 已 有 的 唯一 键 值 的 输入 行 被 跳 过 。 如 果 不 指 定 这 两 个 选项 ， 当 发 现 一 个 复 
制 键 值 时 会 出 现 一 个 错误 ， 并 且 忽 视 文 本 文件 的 剩余 部 分 。 
--silent,-s: 沉默 模式 。 只 有 出 现 错误 时 才 输 出 。 
--socket=path,-S path: 当 连 接 localhost 时 使 用 的 套 接 字 文件 (为 默认 主机 )。 
--Uuser=user_name,-u user_ name: 当 连 接 服务 器 时 MySQL 使 用 的 用 户 名 。 
--verbose,-v: 宛 长 模式 。 打 印 出 程序 操作 的 详细 信息 。 
--Version,-V: 显示 版 本 信息 并 退出 。 
--xmL-X: 产生 XML 输出 。 
mysqldump 提供 许多 选项 ， 包 括 用 于 调试 和 压缩 的 ， 在 这 里 只 是 列举 了 最 有 用 的 。 运 行 
帮助 命令 mysqldump --help， 可 以 获得 特定 版 本 的 完整 选项 列表 。 
Em 如 果 运 行 mysqldump 没有 --quick 或 --opt 选项 ，mysqldump 在 转 储 结果 前 将 整个 
半 ”结果 集 装 入 内 存 。 如 果 转 储 大 数据 库 可 能 会 出 现 问题 。 该 选项 默认 启用 ， 但 可 以 用 
--skip-opt 禁用 。 如 果 使 用 最 新 版 本 的 mysqldump 程序 备份 数据 ， 并 用 于 还 原 到 比较 
旧版 本 的 MySQL 服务 器 中 ， 则 不 要 使 用 --opt 或 -e 选项 。 
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2. 直接 复制 整个 数据 库 目 录 


因为 MySQL 表 保 存 为 文件 方式 ， 所 以 可 以 直接 复制 MySQL 数据 库 的 存储 目录 及 文件 进 
行 备份 。MySQL 的 数据 库 目录 位 置 不 一 定 相 同 ， 在 Windows 平台 下 ，MySQL 5.7 存放 数据 
库 的 目录 通常 默认 为 C:\Documents and Settings\All Users\Application DataMySQL\IMySQL 
Server 5.7\data 或 者 其 他 用 户 自 定义 目录 ; 在 Linux 平台 下 ， 数 据 库 目录 位 置 通常 为 
Awarlib/mysql/， 不 同 Linux 版 本 下 目录 会 有 所 不 同 ， 读 者 应 在 自己 用 的 平台 下 查找 该 目录 。 

这 是 一 种 简单 、 快 速 、 有 效 的 备份 方式 。 要 想 保持 备份 的 一 臻 性， 备份 前 需要 对 相关 表 
执行 LOCK TABLES 操作 ， 然 后 对 表 执 行 FLUSH TABLES。 这 样 当 复制 数据 库 目录 中 的 文件 
时 ， 人 允许 其 他 客户 继续 查询 表 。 需 要 FLUSH TABLES 语句 来 确保 开始 备份 前 将 所 有 激活 的 索 
引 页 写 入 硬盘 。 当 然 ， 也 可 以 停止 MySQL 服务 再 进行 备份 操作 。 

这 种 方法 虽然 简单 ， 但 并 不 是 最 好 的 方法 ， 因 为 这 种 方法 对 InnoDB 存储 引擎 的 表 不 适 
用 。 使 用 这 种 方法 备份 的 数据 最 好 还 原 到 相同 版 本 的 服务 器 中 ， 不 同 的 版 本 可 能 不 兼容 。 


. 
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Sn 在 MySQL 版 本 号 中 ， 第 一 个 数值 表示 主 版 本 号 。 主 版 本 号 相同 的 MySQL 数据 
AS。 库 文件 格式 相同 。 


3. 使 用 mysqlhotcopy 工具 快速 备份 


mysqlhotcopy 是 一 个 Perl 脚本 ， 最 初 由 Tim Bunce 编写 并 提供 。 它 使 用 LOCK 
TABLES、FLUSH TABLES 和 cp 或 scp 来 快速 备份 数据 库 。 它 是 备份 数据 库 或 单个 表 的 最 快 
的 途径 ， 但 它 只 能 运行 在 数据 库 目 录 所 在 的 机 器 上 ， 并 且 只 能 备份 MyISAM 类 型 的 表 。 
mysqlhotcopy 在 Unix 系统 中 运行 。 

mysqlhotcopy 命令 的 语法 格式 如 下 : 


mysqlhotcopy db name 1, ... db name n /path/to/new directory 


db_name 1,.…,db_name n 分 别 为 需要 备份 的 数据 库 的 名 称 ;，/path/to/new_directory 指定 备 
份 文件 目录 。 
【 例 17.18】 使 用 mysqlhotcopy 备份 test 数据 库 到 /usrbackup 目录 下 ， 输 入 语句 如 下 : 


mysqlhotcopy -u root -p test /usr/backup 


要 想 执行 mysqlhotcopy， 必 须 可 以 访问 备份 的 表 文 件 ， 具 有 那些 表 的 SELECT 权限 、 
RELOAD 权限 (以 便 能 够 执行 ELUSH TABLES) 和 LOCK TABLES 权限 。 


Se mysqlhotcopy 只 是 将 表 所 在 的 目录 复制 到 另 一 个 位 置 ， 只 能 用 于 备份 MyISAM 
并 和 ARCHIVE 表 。 备份 InnoDB 类 型 的 数据 表 时 会 出 现 错误 信息 。 由 于 它 复 制 本 地 格 
式 的 文件 ， 故 也 不 能 移植 到 其 他 硬件 或 操作 系统 下 。 


17.6.2 ”数据 还 原 


管理 人 员 操作 的 失误 、 计 算 机 故障 以 及 其 他 意外 情况 ， 都 会 导致 数据 的 丢失 和 破坏 。 当 
数据 丢失 或 意外 破坏 时 ， 可 以 通过 还 原 已 经 备份 的 数据 尽量 减少 数据 丢失 和 破坏 造成 的 损 
失 。 下 面 介绍 数据 还 原 的 方法 。 


1. 使 用 mysql 命令 还 原 


对 于 已 经 备份 的 包含 CREATE、INSERT 语句 的 文本 文件 ， 可 以 使 用 mysql 命令 导入 到 
数据 库 中 。 下 面 介绍 用 mysql 命令 导入 SQL 文件 的 方法 。 

备份 的 SQL 文件 中 包含 CREATE、INSERT 语句 (有 时 也 会 有 DROP 语句 )。mysql 命令 
可 以 直接 执行 文件 中 的 这 些 语句 。 其 语法 格式 如 下 : 


mysql -u user -p [dbname] < filename.sql 


user 是 执行 backup.sql 中 语句 的 用 户 名 ; -p 表示 输入 用 户 密码 ; dbname 是 数据 库 名 。 如 

果 名 ename.sql 文件 为 mysqldump 工具 创建 的 包含 创建 数据 库 语 句 的 文件 ， 则 执行 的 时 候 不 需 
要 指定 数据 库 名 。 

【 例 17.19】 使 用 mysql 命令 将 C:\backup\booksdb 20180301.sql 文件 中 的 备份 导入 到 数据 


库 中 ， 输 入 语句 如 下 : 

mysql -u root -p booksDB < C:/backup/booksdb 20180301.sql 

执行 该 语句 前 ， 必 须 先 在 MySQL 服务 器 中 创建 booksDB 数据 库 ， 如 果 不 存 在 ， 恢 复 过 
程 将 会 出 错 。 命 令 执行 成 功 之 后 ，booksdb 20180301.sql 文件 中 的 语句 就 会 在 指定 的 数据 库 中 
恢复 以 前 的 表 。 

如 果 已 经 登录 MySQL 服务 器 ， 还 可 以 使 用 source 命令 导入 SQL 文件 。source 语句 的 语 
法 如 下 : 

source filename 

【 例 17.20】 使 用 root 用 户 登 录 到 服务 器 ， 然 后 使 用 source 导入 本 地 的 备份 文件 booksDB_ 
20180301.sql， 输 入 语句 如 下 : 

一 选择 要 恢复 到 的 数据 库 


mysql> use booksDB; 
Database changed 


一 -使 用 source 命令 导入 备份 文件 
mysql> source C:\backup\booksDB 20180301.sql 


命令 执行 后 ， 会 列 出 备份 文件 booksDB_20180301.sql 中 每 一 条 语句 的 执行 结果 。source 
命令 执行 成 功 后 ，booksDB_20180301.sql 中 的 语句 会 全 部 导入 到 现 有 数据 库 中 。 
Sn 执行 source 命令 前 ， 必 须 使 用 use 语句 选择 数据 库 。 不然 ， 恢 复 过 程 中 会 出 现 
”ERROR 1046 (3D000): No database selected 的 错误 。 


2. 直接 复制 到 数据 库 目录 


如 果 数 据 库 通过 复制 数据 库 文件 备份 ， 就 可 以 直接 复制 备份 的 文件 到 MySQL 数据 目录 
下 实现 还 原 。 通 过 这 种 方式 还 原 时 ， 必 须 保证 备份 数据 的 数据 库 和 待 还 原 的 数据 库 服务 器 的 
主 版 本 号 相同 。 而 且 这 种 方式 只 对 MyISAM 引擎 的 表 有 效 ， 对 于 InnoDB 引擎 的 表 不 可 用 。 

执行 还 原 以 前 ， 关 闭 mysql 服务 ， 将 备份 的 文件 或 目录 覆盖 MySQL 的 data 目录 ， 启 动 
mysql 服务 。 对 Linux/Unix 操作 系统 来 说 ， 复 制 完 文件 后 ， 需 要 将 文件 的 用 户 和 组 更 改 为 
mysql 运行 的 用 户 和 组 ， 通 常用 户 是 mysql， 组 也 是 mysql。 

3. mysqlhotcopy 快速 恢复 


mysqlhotcopy 备份 后 的 文件 也 可 以 用 来 恢复 数据 库 ， 在 MySQL 服务 器 停止 运行 时 ， 将 备 
份 的 数据 库 文件 复制 到 MySQL 存放 数据 的 位 置 (MySQL 的 data 文件 夹 )， 重 新 启动 MySQL 
服务 即 可 。 如 果 以 根 用 户 执行 该 操作 ， 必 须 指定 数据 库 文件 的 所 有 者 ， 输 入 语句 如 下 : 


chown -R mysql.mysql /var/lib/mysql/dbname 


【 例 17.21】 从 mysqlhotcopy 复制 的 备份 恢复 数据 库 ， 输 入 语句 如 下 : 


cp -R /usr/backup/test usr/local/mysql/data 


执行 完 该 语句 ， 重 启 服务 器 ，MySQL 将 恢复 到 备份 状态 。 
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| 如 果 需 要 恢复 的 数据 库 已 经 存在 ， 则 在 使 用 DROP 语句 删除 已 经 存在 的 数据 库 之 
六 后 ,恢复 才能 成 功 。 另外 ，MySQL 不 同 版 本 之 间 必须 兼容 。 这样 ， 恢 复 之 后 的 数据 
才 可 以 使 用 。 


17.7 疑难 解 惑 
疑问 1: 每 一 个 表 中 都 要 有 一 个 主键 吗 ? 
答 : 并 不 是 每 一 个 表 中 都 需要 主键 ， 一 般 如 果 多 个 表 之 间 进 行 连接 操作 时 ， 需 要 用 到 主 
键 。 因 此 ， 并 不 需要 为 每 个 表 都 建立 主键 ， 而 且 有 些 情 况 最 好 不 使 用 主键 。 
疑问 2: mysqldump 备份 的 文件 只 能 在 MySQL 中 使 用 吗 ? 


答 : mysqldump 备份 的 文本 文件 实际 是 数据 库 的 一 个 副本 。 使 用 该 文件 不 仅 可 以 在 
MySQL 中 恢复 数据 库 ， 而 且 通 过 对 该 文件 进行 简单 修改 ， 还 可 以 在 SQL Server 或 者 Sybase 
等 其 他 数据 库 中 恢复 数据 库 。 这 在 某 种 程度 上 实现 了 数据 库 之 间 的 迁移 。 


疑问 3: 如 何 选择 备份 工具 ? 


答 : 直接 复制 数据 文件 是 最 为 直接 、 快 速 的 备份 方法 ， 但 缺点 是 基本 上 不 能 实现 增 量 备 
份 。 备 份 时 必须 确保 没有 使 用 这 些 表 。 如 果 在 复制 一 个 表 的 同时 服务 器 正在 修改 它 ， 则 复制 
无 效 。 备 份 文件 时 ， 最 好 关闭 服务 器 ， 然 后 重新 启动 服务 器 。 为 了 保证 数据 的 一 致 性 ， 需 要 
在 备份 文件 前 ， 执 行 以 下 SQL 语句 : 


FLUSH TABLES WITH READ LOCK; 


也 就 是 把 内 存 中 的 数据 都 刷新 到 磁盘 中 ， 同 时 锁定 数据 表 ， 以 保证 复制 过 程 中 不 会 有 新 
的 数据 写 入 。 这 种 方法 备份 的 数据 恢复 也 很 简单 ， 直 接 复 制 回 原来 的 数据 库 目 录 下 即 可 。 

mysqlhotcopy 是 一 个 Perl 程序 ， 它 使 用 LOCK TABLES、FLUSH TABLES 和 cp 或 scp 
来 快速 备份 数据 库 。 它 是 备份 数据 库 或 单个 表 的 最 快 途径 ， 但 它 只 能 运行 在 数据 库 文件 所 在 
的 机 器 上 ， 并 且 mysqlhotcopy 只 能 用 于 备份 MyISAM 表 。mysqlhotcopy 适合 于 小 型 数据 库 的 
备份 ， 数 据 量 不 大 ， 可 以 使 用 mysqlhotcopy 程序 每 天 进行 一 次 完全 备份 。 

mysqldump 将 数据 表 导 成 SQL 脚本 文件 ， 在 不 同 的 MySQL 版 本 之 间 升 级 时 相对 比较 合 
适 ， 这 也 是 最 常用 的 备份 方法 。mysqldump 比 直 接 复制 要 慢 一 些 。 
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最 经 典 的 方法 
使 用 MySQLi 操 
作 MySQL 


PHP 是 一 种 简单 的 、 面 向 对 象 的 、 解 释 型 的 、 健 壮 的 、 安 全 的 、 性 能 非常 高 
的 、 独 立 于 架构 的 、 可 移植 的 和 动态 的 脚本 语言 。 而 MySQL 是 快速 且 开 源 的 网 络 
数据 库 系统 。PHP 和 MySQL 的 结合 是 目前 Web 开发 中 的 黄金 组 合 。 

那么 PHP 是 如 何 操作 MySQL 数据 库 的 呢 ? 本 章 介绍 PHP 操作 MySQL 数据 库 
的 各 种 函数 和 技巧 。 
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18.1 PHP 访问 MySQL 数据 库 的 一 般 步 骤 


对 于 一 个 通过 Web 访问 数据 库 的 工作 过 程 ， 一 般 分 为 如 下 几 个 步骤 。 
(1) 用 户 使 用 浏览 器 对 某 个 页 面 发 出 HTTP 请 求 。 
(2) 服务 器 端 接收 到 请 求 ， 发 送 给 PHP 程序 进行 处 理 。 
(3) PHP 解析 代码 。 在 代码 中 有 连接 MySQL 数据 库 的 命令 和 请 求 特定 数据 库 的 某 些 特 
定数 据 的 SQL 命令 。 根 据 这 些 代码 ，PHP 打开 一 个 与 MySQL 的 连接 ， 并 且 发 送 SQL 命令 到 
MySQL 数据 库 





(4) MySQL 接收 到 SQL 语句 之 后 ， 加 以 执行 。 执 行 完毕 后 返回 执行 结果 到 PHP 程序 。 
(5) PHP 执行 代码 ， 并 根据 MySQL 返回 的 请 求 结果 数据 ， 生 成 特定 格式 的 HTML 文 
件 ， 且 传递 给 浏览 器 。HTML 经 过 浏览 器 泻 染 ， 就 得 到 用 户 请 求 的 展示 结果 。 


18.2 ”连接 数据 库 前 的 准备 工作 


从 PHP 5 版 本 开始 ，PHP 连接 数据 库 的 方法 有 两 种 : MySQLi 和 PDO。 本 章 重点 学 习 
MySQLi 的 使 用 方法 和 技巧 。 用 户 首先 需要 开启 对 MySQLi 的 支持 。 

打开 php.ini 文件 ， 找 到 “;extension=php_mysqli.dll”， 去 掉 该 语句 前 的 分 号 “;”， 如 
图 18-1 所 示 ， 保 存 php.ini 文件 ， 重 新 启动 IS 或 Apache 服务 器 即 可 。 


贺 phpiini - 记事 本 口 X 
文件 (篇 句 (E) 格式 (DO) 查看 (V) 帮助 (H) 


php_ldap. dll . 
hp_mbstring. dl 
if. dll 




















; Must be after mbstring as it depends on it 


ef dll 

_openssl. dll 
ion=php_pdo_firebird. dl1 
extension=php_pdo_mysql. dll 
;extension=php_pdo_oci. dll 
;extension=php_oci8_12c. dll ; Use with Oracle Database 12c Instant Client 
;extension=php_pdo_odbc. dl1l 
‘extension=php_pdo_pgsql. dll 
axtension=php_pdo_sqlite. dll 
‘extension=php_pgsql. dll 
;extension= 2 人 dll 
;extension=php_shmop. 


1 The NMIBS data available in the PHP distribution must be installed. 
;See http://www, php. net/manual/en/snmp. installation. php 
;extension=php_snmp. dll 






18-1 修改 php.ini 文 件 


配置 文件 设置 完成 后 ， 可 以 通过 phpinfo0 函 数 来 检查 是 否 配置 成 功 ， 如 果 显 示 出 的 PHP 
的 环境 配置 信息 中 有 mysqli 的 项 目 ， 就 表示 已 经 开启 了 对 MySQL 数据 库 的 支持 ， 如 图 18-2 
所 示 。 
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mysqli 
Client API library version mysqlnd 5 0 12-dev - 20150407 - Sld: 241ae00989d1995fichbf63d579943635faf9972 
Inactive Persistent Links 0 
Active Links 0 
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图 18-2 PHP 的 环境 配置 页 面 


18.3 PHP 操作 MySQL 数据 库 
下 面 介绍 PHP 操作 MySQL 数据 库 所 使 用 的 各 个 函数 的 含义 和 使 用 方法 。 


18.3.1 连接 MySQL 服务 器 


PHP 是 使 用 mysqli_connectO 函 数 连接 到 MySQL 数据 库 的 。 
mysqli_connect() 函 数 的 语法 格式 如 下 : 
mysqli_connect ('MYSQL 服务 器 地 址 '，' 用 户 名 ' ，' 用 户 密码 ' ，' 要 连接 的 数据 库 名 ' ) 


下 面 的 案例 将 连接 服务 器 localhost。 
【 例 18.1】 连 接 服务 器 localhost( 示 例文 件 ch18\18.1.php)。 


<?php 
$servername = "localhost"; 
$username = "root"; 
Spassword = "123456"; 
// 创建 连接 
$db = mysqli_connect ($servername, $username, $password); 
// 检测 连接 
if (!$db) { 
die ("连接 失败 : " . mysqli_connect error()); 


1 
echo "连接 成 功 "; 


2> 
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程序 运行 结果 如 图 18-3 所 示 。 | 二 oa 
@ -IO ete Po|@ ioclhon 
【案例 剖析 】 | 4 PD SEV ta IRM wa 
尝 接 成 功 

该 案例 就 是 通过 此 函数 连接 到 MySQL 数据 库 并 且 把 
此 连接 生成 的 对 象 传递 给 名 为 $db 的 变量 ， 也 就 是 对 象 
$db 。 其 中 MySQL 服务 器 地 址 为 localhost， 用 户 名 为 
root， 用 户 密码 为 本 环境 root 设 定 密码 123456。 

在 默认 情况 下 ，MySQL 服务 的 端口 号 为 3360。 如 果 采 用 默认 的 端口 号 ， 可 以 不 用 指 
定 ; 如 果 采 用 了 其 他 端口 号 ， 比 如 采用 1066 端口 ， 则 需要 特别 指定 ， 例 如 127.0.0.1:1066 表 
示 MySQL 服务 于 本 地 机 器 的 1066 端口 。 

其 中 localhost 换 成 本 地 地 址 或 者 127.0.0.1， 都 能 实现 同样 的 效果 。 

如 果 用 户 在 连接 服务 器 时 ， 同 时 也 连接 好 默认 的 数据 库 为 test， 可 以 将 如 下 代码 : 


$db = mysqli_connect ($servername, $username, $password); 


修改 如 下 : 


$db = mysqli_connect ($servername, $username, $password,test); 


例 18.1 中 连接 数据 库 的 方式 为 面向 过 程 。 由 于 PHP 是 面向 对 象 的 语言 ， 所 以 也 可 以 用 面 
向 对 象 的 方式 连接 MySQL 数据 库 。 代 码 如 下 : 


<?php 

$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
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18-3 ”连接 服务 器 localhost 








// 创建 连接 


$db = new mysqli($servername, $username, $password); 


// 检测 连接 
if ($db->dbect error) { 
die ("连接 失败 : " . $db->dbect_error); 


echo "连接 成 功 "; 


Ce 


18.3.2 选择 数据 库 


连接 到 服务 器 以 后 ， 就 需要 选择 数据 库 ， 只 有 选择 了 数据 库 ， 才 能 对 数据 表 进 行 相关 的 
操作 。 这 里 需要 使 用 函数 mysqli_select_db0 来 选择 。 它 的 语法 格式 如 下 : 

mysqli_select_db (数据 库 服务 器 连接 对 象 ， 目 标 数据 库 名 ) 

下 面 的 案例 将 连接 服务 器 localhost， 然 后 连接 数据 库 test。 

【 例 18.2】 连 接 数 据 库 test( 示 例文 件 ch18\18.2.php)。 


<?php 
$servername = "localhost"; 


合 27s 


$username = "root"7 
$password = "123456"7 
// 创建 连接 
$db = mysqli connect ($servername, $username, $password); 
mysqli_select dbl($db,'test'); 
// 检测 连接 
if ($db) { 
die ("连接 失败 : " . mysqli_connect error()); 


} 
echo "选择 数据 库 成 功 了 "; 


程序 运行 结果 如 图 18-4 所 示 。 
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图 18-4 连接 数据 库 test 
在 新 的 语句 中 ，mysqli_select_db($db,test) 语 句 确定 了 “数据 库 服务 器 连接 对 象 ” 为 gdb， 
而 “目标 数据 库 名 ”为 test。 
py mysqli_ select_ db0O 函 数 经 常 使 用 在 提前 不 知道 应 该 连接 哪个 数据 库 或 者 要 修改 已 
外。 经 连接 的 默认 数据 库 。 





18.3.3 创建 数据 库 

连接 到 MySQL 服务 器 后 ， 用 户 也 可 以 自己 创建 数据 库 。 使 用 mysqli_query0 函 数 可 以 执 
行 SQL 语句 。 语 法 格式 如 下 : 

mysqli_query (dbection,query); 


其 中 参数 dbection 为 数据 库 连 接 ; 参数 query 为 SQL 语句 。 


下 面 的 案例 将 创建 mytest 数据 库 。 

【 例 18.3】 创 建 mytest 数据 库 (示例 文件 ch18\18.3.php)。 

<?php 

$servername = "localhost"; 

$username = "root"; 

Spassword = "123456"; 

// 创建 连接 

$db = mysqli_connect ($servername, $username, $password); 
// 检测 连接 

if (!$db) { 


die ("连接 失败 : " . mysqli connect error()); 
} 


. 
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// 创建 数据 库 

$sql = "CREATE DATABASE mytest"; 

if (mysqli query($db, $sql)) { 
echo "数据 库 创 建成 功 "; 


} else { 


echo "数据 库 创建 失败 : " - mysqli_error ($db); 


. 
// 关 闭 数据 库 的 连接 
mysqli_close($db) 7 


程序 运行 结果 如 图 18-5 所 示 。 








二 oO x 
@ [加 JJ@ meyocno- P ~ © | @ localhost 
文件 (站 六 (查看 (V) 收藏 (A) 工具 (T) 帮助 (H) 
数据 库 创建 成 功 
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18-5 创建 数据 库 mytest 
由 于 PHP 是 面向 对 象 的 语言 ， 所 以 也 可 以 用 面向 对 象 的 方式 创建 MySQL 数据 库 ， 上 面 





的 案例 代码 修改 如 下 : 
<?php 
$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
// 创建 连接 
$db = new mysqli ($servername, $username, S$password); 
// 检测 连接 


if ($db->dbect error) { 
die ("连接 失败 : " . $db->dbect_error); 
i 


// 创建 数据 库 

$sql = "CREATE DATABASE mytest"; 

if ($db->query($sql) === TRUE) { 
echo "数据 库 创建 成 功 "; 

} else { 


echo "数据 库 创 建 失败 : " . $db->error; 
} 


$db->close(); 
2 


| 如 果 服 务 器 端口 不 是 3306， 可 以 指定 自 定义 的 端口 。 例 如 指定 端口 为 3307， 命 
令 如 下 : 


$db = new mysqli($servername, $username, $password,3307); 


18.3.4 创建 数据 表 


数据 库 创建 完成 后 ， 即 可 在 该 数据 库 中 创建 数据 表 。 创 建 数据 表 需 要 执行 CREATE 
TABLE 语句 。 前 面 章节 中 详细 讲述 了 该 语句 的 使 用 方法 ， 这 里 不 再 缆 述 。 
下 面 将 创建 一 个 数据 表 employee， 包 含 4 个 字段 ，SQL 语句 如 下 : 


CREATE TABLE employee 
( 

1d INT(11), 

name VARCHAR(25), 
age INT(4), 

salary FLOAT 

); 


【 例 18.4】 创 建 数据 表 employee( 示 例文 件 ch18\18.4.php)。 


<?php 

$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mytest"; 


// 创建 连接 


$db = mysqli_connect ($servername, $username, $password, $dbname); 


// 检测 连接 
if (!$db) { 

die ("连接 失败 : " . mysqli_connect _ error()); 
;; 


// 使 用 sql 创建 数据 表 
$sql = "CREATE TABLE employee 
( 


id INT (11) UNSIGNED AUTO_INCREMENT PRIMARY KEY, 


name VARCHAR (25) NOT NULL, 
age INT(4) NOT NULL, 
salary FLOAT 
Ne 
if (mysqli_query($db, $sql)) { 
echo "数据 表 employee 创建 成 功 "; 
} else { 
echo "创建 数据 表 错误 : " . mysqli_error ($db); 
} 


mysqli_close ($db); 
?> 


程序 运行 结果 如 图 18-6 所 示 。 
由 于 PHP 是 面向 对 象 的 语言 ， 所 以 也 可 以 用 面向 对 象 的 
方式 创建 MySQL 数据 表 ， 上 面 的 案例 代码 修改 如 下 : 


<?php 
$servername = "localhost"; 
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$username = "root"; 
$password = "123456"7 
$dbname = "mytest"; 


// 创建 连接 
$db = new mysqli ($servername, $username, S$password, $dbname); 
// 检测 连接 
if ($db->connect error) { 
die ("连接 失败 : " . $db->connect_error); 
有 


// 使 用 sql 创建 数据 表 
$sql = " CREATE TABLE employee 
( 





id INT(11) UNSIGNED AUTO_INCREMENT PRIMARY KEY, 
name VARCHAR (25) NOT NULL, 
age INT(4) NOT NULL, 


salary FLOAT 


Fs 


if ($db->query($sql) === TRUE) { 
echo "数据 表 employee 创建 成 功 "; 
} else { 


echo "创建 数据 表 错误 : " . $db->error; 
! 


$db->close (); 
> 


18.3.5 ”添加 数据 


数据 表 创 建 完 成 后 ， 就 可 以 向 表 中 添加 数据 。 需 要 注意 的 是 ， 添 加 数据 的 过 程 中 需要 遵 
循 以 下 原则 。 

(1) PHP 中 SQL 查询 语句 必须 使 用 引号 。 

(2) 在 SQL 查询 语句 中 的 字符 串 值 必须 加 引号 。 

(3) 数值 的 具体 值 不 需要 引号 。 

(4) NULL 值 不 需要 引号 。 

向 MySQL 表 添 加 新 的 记录 需要 使 用 INSERT INTO 语句 。 在 前 面 章 节 中 详细 讲述 了 该 语 
句 的 使 用 方法 ， 这 里 不 再 袭 述 。 

【 例 18.5】 插 入 单条 数据 (示例 文件 ch18\18.5.php)。 


<?php 

$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mytest"; 


// 创建 连接 


$db = mysqli connect ($servername, $username, $password, $dbname); 


// 检测 连接 


全 282 


if (!$db) 1{ 
die (" 数 据 库 连接 失败 : " . mysqli_connect error()); 
} 


$sql = "INSERT INTO employee (id,name,age,salary) 
VALUES (1001，' 张 三 ,，32，4680) ”7 


if (mysqli query($db, $sql)) { 

echo "新 记录 插入 成 功 "; 
Feelse 

echo "插入 数据 错误 : ".$sql . "<br>" . mysqli error ($db); 
3 


mysqli_close ($db); 
这 2 


程序 运行 结果 如 图 18-7 所 示 。 
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图 18-7 插入 单条 数据 
由 于 PHP 是 面向 对 象 的 语言 ， 所 以 也 可 以 用 面向 对 象 的 方式 插入 数据 ， 





修改 如 下 : 


<?php 

$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mytest"; 


// 创建 连接 
$db = new mysqli($servername, $username, $password, $dbname); 
// 检测 连接 
if ($db->connect error) 1{ 
die ("连接 失败 : " . $db->connect_error); 
. 


$sql = "INSERT INTO employee (firstname, lastname, email) 


VALUES (1001, ' 张 三 "， 32, 4680)"; 

if ($db->query($sql) === TRUE) { 
echo "新 记录 插入 成 功 "; 

} else { 


echo "插入 数据 错误 : " . $sql . "<br/>" . $db->error; 
4 


$db->close (); 
2 


上 面 的 案例 代码 
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18.3.6 一 次 插入 多 条 数据 
如 果 一 次 性 想 插入 多 条 数据 ， 需 要 使 用 mysqli_multi query0 函 数 ， 语 法 格式 如 下 : 


mysqli multi_ query (dbection,query); 


其 中 参数 dbection 为 数据 库 连接 ; 参数 query 为 SQL 语句 ， 多 个 语句 之 间 必 须 用 分 号 
隔 开 。 

下 面 的 案例 将 一 次 性 插入 三 条 记录 。 

【 例 18.6】 一 次 插入 三 条 数据 (示例 文件 ch18\18.6.php)。 


<?php 

$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mytest"; 





// 创建 连接 
$db = mysqli_ connect ($servername, $username, $password, $dbname); 
// 检测 连接 
if (!$db) { 

die ("数据 库 连接 失败 : " . mysqli_connect_error()); 
了 
$sql = "INSERT INTO employee (id,name,age,salary) 
VALUES (1002，' 李 四 '，35, 5680);"; 
$sql .= " INSERT INTO employee (id,name,age,salary) 
VALUES (1003， "王蒙 '，32，3680) 7"7 
$sql .= "INSERT INTO employee (id,name,age,salary) 
VALUES (1004, '" 刘 飞 !"，39，6680) "7 


if (mysqli multi query($db, $sql)) { 

echo "三 条 记录 插入 成 功 "; 

3 else 

echo "插入 数据 错误 : ".$sql . "<br>" . mysqli_error ($db); 
} 


mysqli_close ($db); 
人 


程序 运行 结果 如 图 18-8 所 示 。 
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18-8 一 次 插入 三 条 数据 


由 于 PHP 是 面向 对 象 的 语言 ， 所 以 也 可 以 用 面向 对 象 的 方式 一 次 插入 多 条 数据 ， 上 面 的 
案例 代码 修改 如 下 : 


G2 


<?php 

$servername = "localhost"; 
$username = "root"; 
$password = "123456"7 
$dbname = "mytest"; 


// 创建 连接 
$db = new mysqli($servername, $username, $password, $dbname); 
// 检测 连接 
if ($db->connect error) { 
die ("连接 失败 : " . $db->connect _error); 
. 
$sql = "INSERT INTO employee (id,name,age,salary) 
VALUES (1002,，' 李 四 '，35, 5680);"; 
$sql .= " INSERT INTO employee (id,name,age,salary) 
VALUES (1003， ' 王 蒙 '，32，3680) "7 
$sql .= "INSERT INTO employee (id,name,age,salary) 
VALUES (1004, ' 刘 ', 39, 6680)"; 
if ($db-> multi query ($sql) 
echo "三 条 记录 插入 成 功 "; 
} else { 
echo "插入 数据 错误 : " . $sql . "<br/>" . $db->error; 








TRUE) { 
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AG 


S$db->close () 7 
人 


18.3.7” 读 取 数 据 


插入 完 数据 后 ， 读 者 可 以 读 取 数 据 表 中 的 数据 。 下 面 的 案例 主要 学 习 如 何 读 取 employee 
数据 表 的 记录 。 
【 例 18.7】 读 取 数 据 (示例 文件 ch18\18.7.php)。 


<?php 

$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mytest"; 


// 创建 连接 
$db = mysqli_connect ($servername, $username, $password, $dbname); 
// 检测 连接 
if (!$db) { 

die ("数据 库 连 接 失败 : " . mysqli_connect_error ()) 7 
} 
$sql = "SELECT id,name,age,salary FROM employee"; 
$result = mysqli query ($db, $sql); 
if (mysqli num rows($result) > 0) { 

// 输出 数据 

while($row = mysqli fetch assoc($result)) { 

echo "编号 : " . $row["id"]. ”- 姓名 : ” . $row["name"] .” -年龄 : " . 

$row["age"]." 区 革 才 放 $row["salary"]. "<br/>"; 
eelse 
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echo "没有 输出 结果 "; 
. 
mysqli_ free result ($result); 
mysqli_close ($db); 














区 人 
程序 运行 结果 如 图 18-9 所 示 。 
= A 口 x 
@ [B® hpi//ocalho.. P - © | @ localhost 
文件 上 ”六 得 6 下 看 M)， 必 太夫 A)。 工具 (1 帮助 (H) 
编号 : 1001 - 姓名 : 张 三 年龄 : 32 -工资 : 4680 
编号 : 1002 - 姓名 : 李 四 -年 龄 : 35 -工资 : 5680 
编号 : 1003 - 姓名 : 王蒙 -年 龄 : 32 -工资 : 3680 
编号 : 1004 - 姓名 : 刘 飞 “年龄 : 39 -工资 : 6680 
各 100% ~ 
图 18-9 读 取 数 据 
【案例 剖析 】 


(1) SQL 语句 的 作用 是 从 employee 查询 id、name、age 和 salary 这 4 个 字段 ， 然 后 将 查询 
结果 赋 给 变量 $result。 

(2) 使 用 mysqli_ num _rows($result) 函 数 获取 查询 结果 包含 的 数据 记录 的 条 数 。 

(3) 如 果 返 回 的 是 多 条 数据 ， 函 数 mysqli_fetch_assoc 将 结果 集 放 入 到 关联 数组 ， 然 后 配 
合 while 语句 将 查询 结果 输出 。 

由 于 PHP 是 面向 对 象 的 语言 ， 所 以 也 可 以 用 面向 对 象 的 方式 读 取 数 据 表 中 的 数据 ， 上 面 
的 案例 代码 修改 如 下 : 


<?php 

$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mytest"; 


// 创建 连接 
$db = new mysqli ($servername, $username, $password, $dbname); 
// 检测 连接 
if ($db->connect error) { 

die ("连接 失败 : " . $db->connect_error); 
} 
$sql = "SELECT id,name,age,salary FROM employee"; 
$result = mysqli query ($db, $sql); 
if (mysqli num rows($result) > 0) { 

// 输出 数据 

while($row = mysqli fetch assoc($result)) { 

echo "编号 : ” . $row["id"]. ”- 姓名 : ” . $row["name"].” -年 龄 : " . 

$row["age"]." es Nrouml"anlary el "<be/Ss™y 

} 
} else { 

echo "没有 输出 结果 "; 
} 


$db->close (); 
县 之 


站 


18.3.8 释放 资源 


释放 资源 的 函数 为 mysqli_free_result0， 函 数 的 语法 格式 如 下 : 

mysqli_free_result (SQL 请 求 所 返回 的 数据 库 对 象 ) 

例如 ， 上 一 节 中 18.7.php 文件 中 通过 mysqli_free_result($resulb) 语 句 释放 了 SQL 请 求 返 回 
的 对 象 $result 所 占用 的 资源 。 


18.3.9 关闭 连接 


在 连接 数据 库 时 ， 可 以 使 用 mysqli_connect0 函 数 。 与 之 相对 应 ， 在 完成 了 一 次 对 服务 器 
的 使 用 的 情况 下 ， 需 要 关闭 此 连接 ， 以 免 出 现 对 MySQL 服务 器 中 数据 的 误 操 作 。 一 个 服务 
器 的 连接 也 是 一 个 对 象 型 的 数据 类 型 。 

mysqli_connect() 函 数 的 语法 格式 为 : 

mysqli_connect (需要 关闭 的 数据 库 连接 对 象 ) 

在 18.3.7 小 节 的 18.7.php 文件 中 ，mysqli_close($db) 语 句 关 闭 了 $db 连接 。 
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18.4 ”案例 实战 1 一 一 动态 添加 员工 信息 


下 面 将 讲述 PHP 在 开发 动态 网 页 时 如 何 操作 数据 库 。 这 里 以 动态 添加 员工 信息 为 例 进 行 
讲解 。 

【 例 18.8】 使 用 adatabase 的 user 数据 库 表格 ， 添 加 新 的 用 户 信息 。 

创建 18.8.html 文件 。 代 码 如 下 : 


<!doctype html> 
<html> 
<head> 
<title> 添 加 信息 </title> 
</head> 
<body> 
<h2> 添 加 员工 信息 </h2> 
<form action="18.8.php" method="post"> 

员工 姓名 : 

<input name="username" type="text" size="20"/> <br /> 

员工 年 龄 : 
<input name="age" type="text" size="3"/> <br /> 
村 加 辐 忆 ; 绝 
<input name="salary" type="text" size="6"/> <br /> 
<input name="submit" type="submit" value=" 上 传 数据 "/> 
</form> 
</body> 
</html> 


创建 18.8.php 文件 。 代 码 如 下 : 


大 
27@ 
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<?php 

$username = $ POST['username']; 
$age = $_POST['age']; 

$salary = $_POST['salary']; 


$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mytest"; 


// 创建 连接 
$db = mysqli_ connect ($servername, $username, $password, $dbname); 
// 检测 连接 
if (!$db) { 
die ("数据 库 连接 失败 : " . mysqli_connect_error()); 





1 

$username = addslashes ($username); 
$age = addslashes ($age); 

$salary = addslashes ($salary); 


$q = "INSERT INTO employee( name, age,salary) 
VALUES ('$username', $age, $salary)"; 


if(!mysqli_query ($db, $q)){ 
echo "员工 信息 添加 失败 "; 
}jelseli 

echo "员工 信息 已 经 成 功 添加 "7 
3 

mysqli_close ($db); 

人 


ES 运行 18.8.html， 即 可 输入 员工 的 信息 ， 运 行 结果 如 图 18-10 所 示 。 
EC) 单 击 “ 上 传 数据 ”按钮 ， 页 面 跳 转 至 18.8.php， 并 返回 添加 信息 的 情况 ， 如 
图 18-11 所 示 。 
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添加 员工 信息 OE oO cor 
员工 姓名 : BE 本 |] 文件 (月 ”篇 轧 (E) 查看 (V) 收藏 交 (A 工具 (D) 。 莫 助 (H) 
员工 年 的 : BE 员工 信息 已 经 成 功 添加 
员工 工资 : 

上 传 郝 据 
起 100% = 也 100% ~ 
图 18-10 18.8.html 的 运行 结果 图 18-11 员工 信息 添加 成 功 
【案例 训 析 】 


(1) 18.8.html 文件 中 建立 了 employee 数据 表 中 除 id 外 每 个 字段 的 信息 输入 框 。 
(2) 18.8.php 文件 中 建立 MySQL 连接 。 这 里 的 插入 数据 的 语句 为 : $q = "INSERT INTO 
employee( name, age,salary) VALUES ("$username'.$age.$salary)" 用 于 接收 表单 中 的 数据 信息 。 
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18.5 “案例 实战 2 一 一 动态 查询 数据 信息 


本 案例 讲述 如 何 使 用 SELECT 语句 查询 数据 信息 。 这 里 主要 查询 employee 数据 表 中 指定 
年 龄 的 员工 信息 。 

【 例 18.9】 动 态 查询 数据 信息 。 

ECJDp 创建 18.9.html。 代 码 如 下 : 


<!ldoctype html> 

<html> 

<head> 

<title> 查 询 信息 </title> 

</head> 

<body> 

<h2> 查 询 员 工 信 息 </h2> 

<form action="18.9.php" method="post"> 
选择 员工 年 龄 : 

<select name="age"> 

<option value="31">31 岁 </option> 
<option value="32">32 岁 </option> 
<option value="32">33 岁 </option> 
<option value="34">34 岁 </option> 
<option value="35">35 岁 </option> 
</select><br /> 

<input name="submit"” type="submit" value=" 查 询 员工 信息 "/> 
</form> 

</body> 

</html> 


E572Y 创建 18.9.php 文件 。 代 码 如 下 : 


<?php 
$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mytest"; 
$age = $_POST['age']; 
// 创建 连接 
$db = mysqli_connect ($servername, $username, $password, $dbname); 
// 检测 连接 
if (!$db) { 
die ("数据 库 连 接 失败 : " - mysqli_connect_error()); 
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} 
$sql = "SELECT id,name,age,salary FROM employee WHERE age = '".$age.™'"; 
$result = mysqli query ($db, $sql); 
if (mysqli num rows($result) > 0) { 

// 输出 数据 

while($row = mysqli fetch assoc($result)) { 

echo "编号 : ” . $row["id"]. ”= 姓名 : ” - $row["name"] -” =- 年龄 : " . 

Srzow[nadgen] .mw = 工资 * - $row["salary"]. "<br/>*; 
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} 
} else { 

echo "没有 输出 结果 "; 
mysqli free result ($result); 
mysqli close($db); 
2 


ER 运行 18.9.html， 选 择 员工 的 年 龄 ， 例 如 这 里 选择 32 岁 ， 如 图 18-12 所 示 。 
ES 单 击 “ 查 询 员工 信息 ”按钮 ， 页 面 跳 转译 18.9.php， 如 图 18-13 所 示 ， 查 询 出 所 















































有 年 龄 为 32 岁 的 员工 信息 。 
ep 口 x 
@ [@@-@ rpyocalho. P+ © a 
文件 (F) ”六 辑 (E) 查看 (V) 收藏 夫 (A) 工具 (T) 帮 网” @ [CEC Py ET 昌 A 
由: fiP:Wlocalho… localhost 
查询 员工 信息 文件 (久久 (E) 豆 看 (V) 收 基 夫 和 工具 (T) 者 且 (H) 
a 号 : 1001 - 姓名 : 32 -工资 : 4680 
选 掺 员工 年 内: 和 00- 想到 -我 - 形 : 哲 | 
相亲 员工 信息 国 | 
乒 100% ~ 夺 100% ~ 
图 18-12 选择 员工 的 年 龄 图 18-13 ”查询 员工 信息 


18.6 ”提升 安全 性 一 一 防止 SQL 注入 的 攻击 


所 谓 SQL 注入 ， 就 是 通过 把 SQL 命令 插入 到 Web 表单 提交 或 输入 域名 或 页 面 请 求 的 查 
询 字符 串 ， 最 终 达到 欺骗 服务 器 执行 恶意 的 SQL 命令 。 

PHP 7 中 的 预 处 理 语 句 对 于 防止 MySQL 注入 是 非常 有 用 的 。 预 处 理 语句 用 于 执行 多 个 
相同 的 SQL 语句 ， 并 且 执行 效率 更 高 。 

预 处 理 语句 的 工作 原理 如 下 。 

(1) 创建 SQL 语句 模板 并 发 送 到 数据 库 。 预 留 的 值 使 用 参数 “?” 标 记 。 例 如 : 


INSERT INTO employee (id,name,age,salary)VALUES (VALUES(?, ?, ? , ?) 


(2) 数据 库 解 析 ， 编 译 ， 对 SQL 语句 模板 执行 查询 优化 ， 并 存储 结果 而 不 输出 。 

(3) 最 后 ， 将 应 用 绑 定 的 值 传递 给 参数 (“?” 标 记 )， 数 据 库 执行 语句 。 

相 比 于 直接 执行 SQL 语句 ， 预 处 理 语句 有 两 个 主要 优点 。 

(1) 预 处 理 语句 大 大 减少 了 分 析 时 间 ， 只 做 了 一 次 查询 。 

(2) 绑 定 参数 减少 了 服务 器 带宽 ， 只 需要 发 送 查 询 的 参数 ， 而 不 是 整个 语句 。 

预 处 理 语句 针对 SQL 注入 是 非常 有 用 的 ， 因 为 参数 值 发 送 后 使 用 不 同 的 协议 ， 保 证 了 数 
据 的 合法 性 。 

【 例 18.10】 防 止 SQL 注入 (示例 文件 ch18\18.10.php)。 


<?php 

$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 


使 290 


$dbname = "mytest"; 


// 创建 连接 


$db = new mysqli ($servername, $username, $password, $dbname); 


// 检测 连接 


if ($db->connect error) { 
die ("连接 失败 : " . $db->connect error); 
} 


// 预 处 理 及 绑 定 


$stmt = $db->prepare ("INSERT INTO employee (name,age,salary) VALUES 


(Pr 


$stmt->bind param("sii", $name, $age, $salary); 


// 设置 参数 并 执行 
$name = ' 张 晓 峰 '; 
$age = 33; 
$salary = 4550; 
$stmt->execute(); 


$name = ' 张 菲菲 '; 
$age = 37; 
$salary = 4800; 
$stmt->execute(); 


$name = ' 刘 天 佑 '; 
$age = 38; 
$salary = 4300; 
$stmt->execute(); 


echo "新 记录 插入 成 功 "; 


$stmt->close(); 
$db->close(); 
Te 


程序 运行 结果 如 图 18-14 所 示 。 

【案例 剖析 】 

(1) INSERT INTO employee (name,age,salary) 
VALUES (?, ?, ?) 语 句 中 “? ”号 可 以 替换 为 整 型 、 字 
符 串 、 双 精度 浮 点 型 或 布尔 值 。 

(2) $stmt->bind_param("sii", $name, $age, $salary) 语 
句 绑 定 了 SQL 的 参数 ， 且 指定 数据 库 参数 的 值 。sii 参 
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@ 加 :1@ vp//ocaho.. P ~ © | @ localhost 
文件 (月 骗 圭 (5 查看 (V) 收藏 夫 (A) 工具 (T) ”帮助 (H) 
新 记录 插入 成 功 
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18-14 ”防止 SQL 注入 


数列 处 理 其 余 参 数 的 数据 类 型 。 s(string) 字 符 告诉 数据 库 该 参数 为 字符 串 。i(integer) 字 符 告诉 


(3) 每 个 参数 都 需要 指定 类 型 。 通 过 告诉 数据 库 参数 的 数据 类 型 ， 可 以 降低 SQL 注入 的 


风险 。 


. 
219 
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18.7 疑难 解 惑 


疑问 1: 修改 php.ini 文 件 后 仍然 不 能 调用 MySQL 数据 库 怎 么 办 ? 


答 : 有 时 修改 php.ini 文件 不 能 保证 一 定 可 以 加 载 MySQL 函数 库 。 此 时 ， 如 果 使 用 
phpinfo0) 函 数 不 能 显示 MySQL 的 信息 ， 说 明 配置 失败 了 。 重 新 按照 18.2 节 的 内 容 检查 配置 
是 否 正 确 ， 如 果 正 确 ， 则 把 PHP 安装 目录 下 的 libmysql.dll 库 文件 直接 复制 ， 然 后 拷贝 到 系统 
的 system32 目录 下 ， 然 后 重新 启动 IIS 或 Apache， 最 好 再 次 使 用 phpinfo0 进 行 验证 ， 即 可 看 
到 MySQL 信息 ， 表 示 此 时 已 经 配置 成 功 。 

疑问 2: 为 什么 应 尽量 省 略 MySQL 语句 中 的 分 号 ? 

答 : 在 MySQL 语句 中 ， 每 一 行 的 命令 都 是 用 分 号 (:) 作 为 结束 的 。 但 是 ， 当 一 行 MySQL 
被 插入 到 PHP 代码 中 时 ， 最 好 把 后 面 的 分 号 省 略 掉 。 这 主要 是 因为 PHP 也 是 以 分 号 作为 一 行 
的 结束 的 ， 额 外 的 分 号 有 时 会 让 PHP 的 语法 分 析 器 搞 不 明白 ， 所 以 还 是 省 略 掉 为 好 。 在 这 种 
情况 下 ， 虽 然 省 略 了 分 号 ， 但 是 PHP 在 执行 MySQL 命令 时 会 自动 加 上 去 的 。 

另外 ， 还 有 一 个 不 要 加 分 号 的 情况 。 当 用 户 想 把 字段 竖 着 排列 显示 下 来 ， 而 不 是 像 通常 
的 那样 横着 排列 时 ， 可 以 用 G 来 结束 一 行 SQL 语句 ， 这 时 就 用 不 上 分 号 了 ， 例 如 : 

SELECT * FROM paper WHERE USER_ID 三 1G 

疑问 3: 如 何 对 数据 表 中 的 信息 进行 排序 操作 ? 

答 : 使 用 ORDER BY 语句 可 以 对 数据 表 中 的 信息 进行 排序 操作 。 例 如 ， 将 数据 表 
employee 中 的 信息 按 年 龄 从 小 到 大 排序 。SQL 语句 如 下 : 


SELECT id,name age salarYy FROM employee ORDER BY age RSC 


其 中 ASC 为 默认 关键 词 ， 表 示 按 升序 排列 。 如 果 想 降序 排列 ， 可 以 使 用 DESC 关键 词 。 


第 19 章 
最 兼容 的 方法 
使 用 PDO 操作 
MySQL 数据 库 





PHP 的 数据 库 抽 象 类 的 出 现 是 PHP 发 展 过 程 中 重要 的 一 步 。PDO 扩展 为 PHP 
访问 数据 库 定义 了 一 个 轻 量 级 的 、 一 致 性 的 接口 ， 它 提供 了 一 个 数据 访问 抽象 层 。 
这 样 ， 无 论 使 用 什么 数据 库 ， 都 可 以 通过 一 致 的 函数 执行 查询 和 获取 数据 。 本 章 主 
要 介绍 PDO 数据 库 抽 象 类 库 的 使 用 方法 。 
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19.1 认识 PDO 


随 着 PHP 应 用 的 快速 增长 和 通过 PHP 开发 跨 平台 应 用 的 普及 ， 使 用 不 同 的 数据 库 是 十 分 
常见 的 。PHP 需要 支持 MySQL、SQL Server 和 Oracle 等 多 种 数据 库 。 

如 果 只 是 通过 单一 的 接口 针对 单一 的 数据 库 编 写 程序 ， 比 如 用 MySQL 函数 处 理 MySQL 
数据 库 ， 用 其 他 函数 处 理 Oracle 数据 库 ， 这 在 很 大 程度 上 增加 了 PHP 程序 在 数据 库 方面 的 灵 
活性 并 提高 了 编程 的 复杂 性 和 工程 量 。 

如 果 通 过 PHP 开发 一 个 跨 数 据 库 平台 的 应 用 ， 比 如 对 于 一 类 数据 需要 到 两 个 不 同 的 数据 
库 中 提取 数据 ， 在 使 用 传统 方法 的 情况 下 只 好 写 两 个 不 同 的 数据 库 连 接 程 序 ， 并 且 要 对 两 个 
数据 库 连 接 的 工作 过 程 进行 协调 。 

为 了 解决 这 个 问题 ， 程 序 员 们 开发 出 了 “数据 库 抽象 层 ”。 通 过 这 个 抽象 层 ， 把 数据 处 
理 业 务 逻 辑 和 数据 库 连接 区 分 开 来 。 也 就 是 说 ， 不 管 PHP 连接 的 是 什么 数据 库 ， 都 不 影响 
PHP 程序 的 业务 逻辑 。 这 样 对 一 个 应 用 来 说 ， 就 可 以 采用 若干 个 不 同 的 数据 库 支持 方案 。 

PDO 就 是 PHP 中 最 为 主流 的 实现 “数据 库 抽 象 层 ”的 数据 库 抽象 类 。PDO 类 是 PHP 中 
最 为 突出 的 功能 之 一 。 在 PHP 5 版 本 以 前 ，PHP 都 是 只 能 通过 针对 MySQL 的 类 库 、 针 对 
Oracle 的 类 库 、 针 对 SQL Server 的 类 库 等 实现 有 针对 性 的 数据 库 连 接 。 

PDO 是 PHP Data Objects 的 简称 ， 是 为 PHP 访问 数据 库 定 义 的 一 个 轻 量 级 的 、 一 致 性 的 
接口 ， 它 提供 了 一 个 数据 访问 抽象 层 。 这 样 ， 无 论 使 用 什么 数据 库 ， 都 可 以 通过 一 致 的 函数 
执行 查询 和 获取 数据 。 

PDO 通过 数据 库 抽象 层 实现 了 以 下 一 些 特性 。 

(1) 灵活 性 。 可 以 在 PHP 运行 期 间 ， 直 接 加 载 新 的 数据 库 ， 而 不 需要 在 新 的 数据 库 使 用 
时 ， 重 新 设置 和 编译 。 

(2) 面向 对 象 。 这 个 特性 完全 配合 了 PHP， 通 过 对 象 来 控制 数据 库 的 使 用 。 

(3) 速度 极 快 。 由 于 PDO 是 使 用 C 语言 编写 并 且 编 译 进 PHP 的 ， 所 以 比 那 些 用 PHP 编 
写 的 抽象 类 要 快 很 多 。 








19.2 PDO 的 安装 


由 于 PDO 类 库 是 PHP 7 自 带 的 类 库 ， 所 以 要 使 用 PDO 类 库 ， 只 需要 在 php.ini 中 把 关于 
PDO 类 库 的 语句 前 面 的 注释 符号 去 掉 。 

首先 启用 extension=php_pdo.dll 类 库 ， 这 个 类 库 是 PDO 类 库 本 身 。 然 后 是 不 同 的 数据 库 
驱动 类 库 选 项 。extension=php_pdo_ mysql.dll 适用 于 MySQL 数据 库 的 连接 。 如 果 使 用 SQL 
Server， 可 以 启用 extension=php_pdo_mssql.dll 类 库 。 如 果 使 用 Oracle 数据 库 ， 可 以 启用 
extension=php_pdo_oci.dll。 除 了 这 些 ， 还 有 支持 PgSQL 和 SQLite 等 的 类 库 。 

本 机 环境 下 启用 的 类 库 为 extension=php_pdo.dll 类 库 和 extension=php_pdo mysqldll 类 
库 ， 如 图 19-1 所 示 。 


司 phpin -is 二 一 o x 
文件 (六福 格式 (O) 查看 (W) 帮助 (H) 


extensi 
extensi mbstring. dll 
exif 
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si on Php oci8_l2c. 1 1 Use with Oracle Database 12c Instant 
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iextension=php_pdo_odbc. dl1 

iextension=php_pdo_pgsql.dll 
extension=pho_pdo_sqlite. dll 

Textension=php_pgsql. dll 

Textension=php_phpdbg_wehhelper. dll 

;extension=php_shmop. dll 


1 The MIBS data ayailable in the PHP distribution must be installed. 
1 See http: /ww. Php. net/manual /en/snmp. installation. php 
‘extension=} dll 


extension=php_soap. dll 
extension=php_sockets. dl1 
extension=php_sqlite3.dll 所 








19-1 配置 php.ini 文件 
可 以 通过 phpinfo() 查看 是 否 安装 成 功 ， 如 图 19-2 所 示 。 
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图 19-2 查看 安装 是 否 成 功 


19.3 ”使 用 PDO 操作 MySQL 


在 本 开发 环境 下 使 用 的 数据 库 是 MySQL， 所 以 在 使 用 PDO 操作 数据 库 之 前 ， 需 要 首先 
连接 到 MySQL 服务 器 和 特定 的 MySQL 数据 库 。 
这 个 操作 是 通过 PDO 类 库 内 部 的 构造 函数 来 完成 的 。PDO 的 构造 函数 的 语法 格式 如 下 : 


PDO: :constuct (DSN, username, password, driver options) 

其 中 DSN 是 一 个 “数据 源 名 称 ”，username 是 接 入 数据 源 的 用 户 名 ，password 是 用 户 密 
码 ，driver_options 是 特定 连接 要 求 的 其 他 参数 。 

DSN 是 一 个 字符 串 ， 它 是 由 “数据 库 服务 器 类 型 ”“ 数 据 库 服务 器 地 址 ”和 “数据 库 名 
称 ” 组 成 的 。 它 们 组 合 的 语法 格式 如 下 : 

"数据 库 服务 器 类 型 :host= 数 据 库 服务 器 地 址 ;dbname= 数 据 库 名 称 ' 

driver_options 是 一 个 数组 ， 它 有 很 多 选项 。 
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@ PDO::ATTR AUTOCOMMIT: 此 选项 定义 PDO 在 执行 时 是 否 注 释 每 条 请 求 。 

e@ PDO::ATTR_CASE: 通过 此 选项 ， 可 以 控制 在 数据 库 中 取得 的 数据 的 字母 的 大 小 
写 。 具 体 说 来 就 是 ， 可 以 通过 PDO::CASE UPPER 使 所 有 读 取 的 数据 字母 变 为 大 
写 ; 可 以 通过 PDO::CASE LOWER 使 所 有 读 取 的 数据 字母 变 为 小 写 ， 可 以 通过 
PDO::CASE_NATURL 使 用 特定 的 在 数据 库 中 发 现 的 字段 。 

@ 。 PDO::ATTR_EMULATE PREPARES: 此 选项 可 以 利用 MySQL 的 请 求 缓存 功能 。 

@ PDO::ATTR_ERRMODE: 使 用 此 选项 定义 PDO 的 错误 报告 模型 。 具 体 的 3 种 模式 
分 别 为 PDO::ERRMODE_EXCEPTION 异常 模式 、PDO::ERRMODE _SILENT 沉默 模 
式 和 PDO::ERRMODE WARNING 警报 模式 。 

@ PDO::ATTR_ ORACLE NULLS: 此 选项 在 使 用 Oracle 数据 库 时 会 把 空 字 符 串 转换 为 
NULL 值 。 一 般 情 况 下 ， 此 选项 默认 为 关闭 。 

@ 。 PDO::ATTR PERSISTENT: 使 用 此 选项 来 确定 此 数据 库 连 接 是 否 可 持续 。 但 是 其 默 
认 值 为 false， 不 启用 。 

@ PDO::ATTR_PREFETCH: 此 选项 确定 是 否 要 使 用 数据 库 的 prefetch 功能 。 此 功能 是 
在 用 户 取得 一 条 记录 操作 之 前 就 取得 多 条 记录 ， 以 准备 给 其 下 一 次 请 求 数据 操作 提 
供 数据 ， 并 且 减 少 了 执行 数据 库 请 求 的 次 数 ， 提 高 了 效率 。 

e@ PDO::ATTR_TIMEOUT: 此 选项 设置 超时 时 间 的 秒 数 。 但 MySQL 不 支持 此 功能 。 

@ PDO::DEFAULT FETCH MODE: 此 选项 可 以 设 定 默 认 的 fetch 模型 ， 是 以 联合 数据 
的 形式 取得 数据 ， 或 以 数字 索引 数组 的 形式 取得 数据 ， 或 以 对 象 的 形式 取得 数据 。 


19.3.1 连接 MySQL 数据 库 


当 建 立 一 个 连接 对 象 的 时 候 ， 只 需要 使 用 new 关键 字 ， 生 成 一 个 PDO 的 数据 库 连接 实例 
即 可 。 
例如 ， 使 用 MySQL 作为 数据 库 生 成 一 个 数据 库 连 接 。 代 码 如 下 : 


$db = new PDO('mysql:host=localhost;dbname=mytest','root','123456'); 


如 果 连 接 数 据 库 有 错误 ， 将 会 抛 出 一 个 PDOException 异常 对 象 。 

用 户 可 以 使 用 ty-catch 异常 处 理 机 制 。 代 码 如 下 : 

<?php 

try { 

$dbconnect = 

new PDO('mysql:host=localhost;dbname=mytest','root','123456'); 
} catch (PDOException $exception) { 

echo "数据 库 连 接 错 误 : "” . $exception->getMessage(); 


如 果 连 接 数据 库 成 功 ， 返 回 一 个 PDO 类 的 实例 给 脚本 ， 此 连接 在 PDO 对 象 的 生存 周期 
中 保持 活动 。 

要 想 关 闭 连接 ， 需 要 销毁 对 象 以 确保 所 有 剩余 对 它 的 引用 都 被 删除 ， 可 以 赋 一 个 NULL 
值 给 对 象 变量 。 
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如 果 不 销毁 对 象 ，PHP 在 程序 结束 时 会 自动 关闭 连接 。 

关闭 一 个 连接 的 命令 : 

<?php 

$db = new PDO('mysql:host=localhost;dbname=mytest','root','123456'); 


// 在 此 使 用 连接 


// 现在 运行 完成 ， 在 此 关闭 连接 

$db = null; 

汪汪 

但 是 ， 很 多 Web 应 用 程序 却 需要 持久 性 的 连接 ， 因 为 持久 性 连接 在 程序 结束 后 不 会 被 关 
闭 ， 且 被 缓存 ， 当 另 一 个 使 用 相同 凭证 的 程序 连接 请 求 时 被 重用 。 可 见 ， 持 久 连 接 缓存 可 以 
避免 每 次 程序 都 需要 建立 一 个 新 连接 的 开销 ， 从 而 让 Web 应 用 程序 更 快 。 

设置 持久 化 连接 的 命令 : 

<?php 

$db = new PDO('mysql:host=localhost;dbname=mytest','root','123456',array( 

PDO: :ATTR_PERSISTENT => true 
)); 
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人 
@: 如 果 想 使 用 持久 连接 ， 必 须 在 传递 给 PDO 构造 函数 的 驱动 选项 数组 中 设置 
和 PDO::ATTR_PERSISTENT. 


pp A 


19.3.2 ”创建 数据 库 
连接 到 MySQL 服务 器 后 ， 用 户 也 可 以 自己 创建 数据 库 。 例 如 ， 下 面 的 案例 将 创建 


mypdo 数据 库 。 
【 例 19.1】 创 建 mypdo 数据 库 (示例 文件 ch19\19.1.php)。 
<?php 
$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
try { 


$db = new PDO("mysql:host=$servername;dbname=mytest", $username, 
$password); 


// 设置 PDo 错误 模式 为 异常 
$db->setAttribute (PDO: :ATTR_ERRMODE, PDO::ERRMODE EXCEPTION) ; 
$sql = "CREATE DATABASE mypdo"; 


// 使 用 exec () ， 因 为 没有 结果 返回 
$db->exec ($sql); 


echo "数据 库 创 建成 功 <zbr/>"; 


catch (PDOException $e) 
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{ 
echo $sql - "<br>" . $e->getMessage(); 
y 


$db = null; 
?> 
程序 运行 结果 如 图 19-3 所 示 。 
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数据 库 创建 成 功 
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19-3 ”创建 数据 库 mypdo 





Dr 使 用 PDO 的 最 大 好 处 是 在 数据 库 查 询 过 程 出 现 问题 时 可 以 使 用 异常 类 来 处 理 问 
慌 题 。 如 果 try{ } 代码 块 出 现 异常 ， 脚 本 会 停止 执行 并 会 跳 到 第 一 个 catchO{ } 代 码 块 


执行 代码 。 在 上 面 的 捕获 的 代码 块 中 输出 了 SQL 语句 并 生成 错误 信息 。 


19.3.3 创建 数据 表 


数据 库 创 建 完成 后 ， 即 可 在 该 数据 库 中 创建 数据 表 。 创 建 数据 表 需 要 执行 CREATE 
TABLE 语句 。 
下 面 创建 一 个 数据 表 fruits， 包 含 4 个 字段 ，SQL 语句 如 下 : 


CREATE TABLE fruits 
( 

id INT(11), 
name VARCHAR(20), 
amount INT(8) 
salary FLOAT 

); 


【 例 19.2】 创 建 数 据 表 fruits( 示 例文 件 ch19\19.2.php)。 


<?php 

$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mypdo"; 


try { 
$db = new PDO("mysql:host=$servername;dbname=$dbname", $username, 
$password); 


// 设置 PDo 错误 模式 ， 用 于 抛 出 异常 


$db->setAttribute (PDO: :ATTR_ERRMODE, PDO: :ERRMODE_EXCEPTION) 


// 使 用 sql 创建 数据 表 


$sql = "CREATE TABLE fruits( 
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“| 
第 中 
id INT(6) UNSIGNED AUTO_ INCREMENT PRIMARY KEY, 六 
name VARCHAR (20) NOT NULL, 章 
amount INT(8) NOT NULL, 
salary FLOAT 最 
| 
// 使 用 exec () ， 没 有 结果 返回 的 
$db->exec ($sql); 法 
echo "数据 表 fruits 创建 成 功 "; 
catch (PDOException $e) 
二 刁 
echo $sql . "<br/>" . $e->getMessage(); 9 
} 操 
作 
$db = null; 三 
> 了 
2 
程序 运行 结果 如 图 19-4 所 示 。 a 数 
@ @ pliocho- P ~ © | @ localhot 据 
村 文件 (有 篇 纺 (查看 (V) 收藏 只 A】 工具 (大助 (H) 库 
19.3.4 添加 数据 数据 表 fruits 创 建成 功 
数据 表 创 建 完成 后 ， 就 可 以 向 表 中 添加 数据 。 向 io 





MySQL 表 添 加 新 的 记录 需要 使 用 INSERT INTO 语 
句 ， 在 前 面 章节 中 详细 讲述 了 该 语句 的 使 用 方法 ， 这 
里 不 再 殉 述 。 

【 例 19.3】 插 入 单条 数据 (示例 文件 ch19\19.3.php)。 

<?php 

$servername = "localhost"; 

$username = "root"; 


$password = "123456"; 
$dbname = "mypdo"; 


19-4 创建 数据 表 fruits 





try { 

$db = new PDO("mysql:host=$servername;dbname=$dbname", $username, 
$password); 

// 设置 PDo 错误 模式 ， 用 于 抛 出 异常 

$db->setAttribute (PDO: :ATTR_ERRMODE, PDO::ERRMODE EXCEPTION); 
$sql = "INSERT INTO fruits (name, amount, salary) 

VALUES (' 苞 果 '，3000，3.2)"; 

// 使 用 exec () ， 没 有 结果 返回 

$db->exec ($sql); 

echo "新 记录 插入 成 功 "7 
} 
catch (PDOException $e) 

echo $sql . "<br/>" . $e->getMessage(); 
. 


$db = null; 
2> 
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程序 运行 结果 如 图 19-5 所 示 。 2 又 
@ -© reocsho- P- © |@ iccalhost 


Eu 


19.3.5 一 次 插入 多 条 数据 入 成功 











如 果 一 次 性 想 插入 多 条 数据 ， 需 要 通过 事务 来 完成 。 下面。 一 一 一 一 一 一 一 一 
的 案例 将 一 次 性 插入 三 条 记录 。 图 19-5 “插入 单条 数据 

【 例 19.4】 一 次 插入 三 条 记录 (示例 文件 ch19\19.4.php)。 

<?php 





$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mypdo"7 


try { 
$db = new PDO("mysql:host=$servername;dbname=$dbname", $username, 
$password); 
// 设置 PDo 错误 模式 ， 用 于 抛 出 异常 
$db->setAttribute (PDO: :ATTR_ERRMODE, PDO::ERRMODE EXCEPTION); 
// 开始 事务 
$db->beginTransaction(); 
// SQL 语句 
$db->exec ("INSERT INTO fruits (name, amount, salary) 
VALUES (' 香 燕 '，3200，3.6)"); 
$db->exec ("INSERT INTO fruits (name, amount, salary) 
VALUES ('" 橘 子 '，3300，3.8)") 7 
$db->exec ("INSERT INTO fruits (name, amount, salary) 
Was R00 SD 
// 提交 事务 
$db->commit (); 
echo "三 条 记录 插入 成 功 "; 
catch (PDOException $e) 


// 如 果 执 行 失败 回 深 

$db->rollback (); 

echo $sql . "<br/>" . $e->getMessage(); 
， 





$db = null; 

> 

程序 运行 结果 如 图 19-6 所 示 。 a -0 x 
@ [由 -|@@ herylccaho.P - 上 | @ lecalhost 








文件 中 。 综 熏 (日 坦 看/) 。 收 殴 夫人 A) 工具 Mm 者 动 HH) 


19.3.6 ” 读 取 数据 三 条 i 录 括 入 成 内 








插入 完 数据 后 ， 读 者 可 以 读 取 数 据 表 中 的 数据 。 下 面 Sis 
的 案例 主要 学 习 如 何 读 取 fruits 数据 表 中 的 记录 ， 并 将 记 。 。 图 19.6 一 次 插入 三 条 数据 
录 显 示 在 表格 中 。 


二 
G3 


“uu 


【 例 19.5】 读 取 数 据 (示例 文件 ch19\19.5.php)。 


<?php 
echo "<table style='border: solid lpx black;'>"; 
echo "<tr><th> 编 号 </th><th> 名 称 </th><th> 数 量 </th><th> 单 价 </th></tr>"; 


class TableRows extends RecursiVeIteratorIterator { 
function _ construct ($it) { 
parent::_ construct ($it, self::LEAVES ONLY); 


function current() { 
return “<td style='width:150px;border:lpx solid black;'>" . parent:: 
current(). “</td>"s 





function beginChildren() { 
echo., "<tr>"y 
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function endchildren() { 
EDGE 二 EN 
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$servername = "localhost"; 
$username = "root"; 
$password = "123456"; 
$dbname = "mypdo"; 


try { 
$db = new PDO("mysql:host=$servername;dbname=$dbname", $username, S$password); 
$db->setAttribute (PDO: :ATTR_ERRMODE, PDO::ERRMODE EXCEPTION); 
$stmt = $db->prepare ("SELECT id, name, amount, salary FROM fruits"); 
$stmt->execute () 7 


// 设置 结果 集 为 关联 数组 

$result = $stmt->setFetchMode (PDO: :FETCH ASSOC); 

foreach (new TableRows (new RecursiveArrayIterator($stmt->fetchAll())) as 
$k=>$v) { 
echo $v; 


} 
catch (PDOException Se) { 

echo " 读 取 数 据 错误 : " . $e->getMessage (); 
} 


$db = null; 

echo "</table>"; 

?> 

程序 运行 结果 如 图 19-7 所 示 。 
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19-7” 读 取 数 据 


19.4 ”提升 安全 性 一 一 防止 SQL 注入 的 攻击 


PHP 7 中 的 预 处 理 语句 对 于 防止 MySQL 注入 是 非常 有 用 的 。 当 执行 一 个 SQL 语句 时 ， 
需要 PDO 对 语句 进行 执行 。 正 常情 况 下 可 以 逐 句 执行 。 而 每 执行 这 样 一 句 ， 都 需要 PDO 首 
先 对 语句 进行 解析 ， 然 后 传递 给 MySQL 来 执行 。 这 都 需要 PDO 的 工作 。 如 果 是 不 同 的 SQL 
语句 ， 则 这 是 必要 过 程 。 但 如 果 是 INSERT 这 样 的 语句 ， 语 句 结构 都 一 样 ， 只 是 每 一 项 具体 
的 数值 不 同 ， 在 这 种 情况 下 PDO 的 prepare 表述 就 可 以 只 提供 改变 的 变量 值 ， 而 不 改变 SQL 
语句 ， 起 到 减少 解析 过 程 、 节 省 资源 、 提 高 效率 和 防止 SQL 注入 的 作用 。 

【 例 19.6】 防 止 SQL 注入 (示例 文件 ch19\19.6.php)。 

<?php 

$servername = "localhost"; 

$username = "root"; 


$password = "123456"; 
$dbname = "mypdo"; 





try { 

$db = new PDO("mysql:host=$servername;dbname=$dbname", $username, 
$password); 

// 设置 PDO 错误 模式 为 异常 


$db->setAttribute (PDO: :ATTR_ERRMODE, PDO::ERRMODE EXCEPTION); 


// 预 处 理 SQL 并 绑 定 参数 

$stmt = $db->prepare ("INSERT INTO fruits (name, amount, salary) VALUES 
(:name, :amount, :salary)"); 

$stmt->bindParam(':name', $name); 

$stmt->bindParam(':amount', $amount, PDO::PARAM INT); 

$stmt->bindParam(':salary', $salary, PDO::PARAM STR); 


// 插入 第 一 行 

$name = " 检 子 "; 
$amount = 2300; 
$salary = 3.9; 


$stmt->execute(); 


// 插入 第 二 行 
Sname = "火龙 果 "; 
$amount = 2600; 
$salary = 5.2; 
$stmt—->execute(); 


// 插入 第 三 行 
$name = “" 险 密 瓜 "; 
$amount = 3600; 
$salary = 3.8; 
$stmt->execute (); 


echo "三 行 记录 插入 成 功 "; 





catch (PDOException $e) 
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-_ OO x 
文件 (月 ” 编 铝 (E) ” 喜 看 V) 收藏 夫 (A) 工具 (T) 帮助 (H) 











echo "插入 数据 错误 : " . $e->getMessage(); 三 行 记录 插入 成 村 
四 三 
$db = null; | 
2> 轩 100% 7 
程序 运行 结果 如 图 19-8 所 示 。 19-8 防止 SQL 注入 


19.5 疑难 解 惑 


疑问 1: 在 操作 MySQL 数据 库 时 ， PDO 和 MySQLi 到 底 哪 个 好 ? 


答 : PDO 和 MySQLi 各 有 优势 ， 具 体 介 绍 如 下 。 

(1) PDO 应 用 在 12 种 不 同 数据 库 中 ，MySQLi 只 针对 MySQL 数据 库 。 

(2) 两 者 都 是 面向 对 象 ， 但 MySQLi 还 提供 了 API 接口 。 

(3) 两 者 都 支持 预 处 理 语 句 。 预 处 理 语句 可 以 防止 SQL 注入 ， 对 于 Web 项 目的 安全 性 
是 非常 重要 的 。 

可 见 ， 如 果 项 目 需要 在 多 种 数据 库 之 间 切 换 ， 建 议 使 用 PDO， 因 为 只 需要 修改 连接 字符 
串 和 部 分 查询 语句 即 可 。 使 用 MySQLi， 如 果 是 不 同 的 数据 库 ， 需 要 重新 编写 所 有 代码 ， 包 
括 查 询 语句 。 

疑问 2: 在 PDO 中 事务 如 何 处 理 ? 

答 : 在 PDO 中 同样 可 以 实现 事务 处 理 的 功能 ， 有 具体 使 用 方法 如 下 。 

(1) 开启 事务 。 使 用 beginTransaction() 方 法 将 关闭 自动 提交 模式 ， 直 到 事务 提交 或 者 回 
滚 以 后 才 恢 复 。 

(2) 提交 事务 。 使 用 commit() 方 法 完成 事务 的 提交 操作 ， 成 功 则 返回 TRUE， 和 否则 返回 
FALSE。 

(3) 事务 回 深 。 使 用 rollBack( 方 法 执行 事务 的 回 滚 操作 。 

疑问 3: 如 何 通 过 PDO 连接 SQL Server 数据 库 ? 


通过 PDO 可 以 实现 与 SQL Server 数据 库 连接 的 操作 。 下 面 通过 例子 来 讲解 具体 的 连接 方 
法 。 代 码 如 下 : 


<?php 
header ("Content-Type:text/html;charset=utf-8"); // 设 置 页 面 的 编码 风格 
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$host = 'PC-201405212233'; // 设 置 主机 名 称 

$user = 'sa'; // 设 置 用 户 名 

$pwd = '123456'; // 设 置 密 码 

$dbName = 'mydatabase'7 // 设 置 需要 连接 的 数据 库 


$dbms = 'mssql'; // 

$dsn = "mssql:host=$host;dbname-$dbName"; 

try{ // 利 用 try-catch 捕获 异常 
$pdo = new PDO($dsn, $user, $pwd); 

echo "成 功 连接 SQL server 数据 库 " 

}catch (Exception $e){ 

Die ("错误 提示 ! ".$e->getMessage ()) 

上 
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项 目 实 训 1 一 一 开发 验证 码 系统 

项 目 实 训 2 一 一 开发 个 人 博客 系统 
项 目 实 训 3 一 一 开发 用 户 权限 系统 
项 目 实 训 4 一 一 开发 社区 市 场 系统 


第 20 章 
项 目 实 训 1 一 一 
开发 验证 码 系统 


验证 码 系统 在 开发 动态 网 站 中 最 为 常见 。 本 章 介 绍 验证 码 系统 的 开发 方法 。 通 
过 本 章 的 学 习 ， 读 者 可 以 进一步 巩固 前 面 所 学 的 知识 ， 包 括 GD 库 的 使 用 方法 和 
Session 的 管理 技术 。 
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20.1 系统 分 析 
在 开发 验证 码 系统 之 前 ， 首 先 应 分 析 系 统 的 需求 。 


20.1.1 学 习 目标 


此 验证 码 系统 的 前 台 采 用 Bootstrap 和 jQuery 搭建 ， 后 台 基 于 PHP 7 版 本 ， 使 用 GD 库 和 
Session 管理 技术 。 

通过 该 案例 ， 读 者 可 以 实现 以 下 学 习 目标 。 

(1) 学 习 类 的 定义 和 使 用 方法 。 

(2) 学 习 Session 的 特性 和 使 用 场景 。 

(3) 通过 GD 库 生 成 图 片 与 输出 图 片 的 技巧 。 

(4) 进一步 理解 对 象 的 属性 和 方法 的 作用 域 。 

(5) 熟练 使 用 循环 和 分 支 语 句 。 

(6) 学 习 验 证 码 的 开发 思路 。 

(7) 学 习 防 止 用 户 重 复 提 交 或 非 人 为 提交 验证 码 的 方法 和 技巧 。 


20.1.2 ”需求 分 析 


该 项 目 是 一 个 验证 码 生 成 与 检验 系统 ， 主 要 需求 如 下 。 

(1) 生成 验证 码 图 片 并 以 网 页 的 形式 展示 给 用 户 。 

(2) 用 户 输入 验证 码 并 提交 后 ， 检 查 所 提交 的 验证 码 的 正确 性 ， 并 将 检查 结果 显示 给 用 户 。 

(3) 验证 码 的 有 效 期 为 3 分 钟 ， 超 过 3 分 钟 验 证 码 即 失效 ， 需 要 重新 刷新 验证 码 。 并 且 验 
证 码 只 能 使 用 一 次 ， 多 次 使 用 同样 失效 。 


20.1.3 ”系统 文档 
验证 码 系统 的 文档 如 图 20-1 所 示 。 
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图 20-1 系统 文档 
各 个 文档 的 含义 如 下 。 


(1) index.php: 显示 验证 码 图 片 ， 并 提交 检查 用 户 输入 的 验证 码 。 


二 
全 308 


(2) code php: 生成 并 输入 验证 码 图 片 。 

(3) controller/CodeController.php: 验证 码 控制 器 ， 生 成 并 输出 验证 码 图 片 ， 验 证 提交 的 验 
证 码 的 正确 性 。 

(4) css/bootstrap.css: 设置 主页 的 样式 。 

(5) js/jquery-3.2.1.js: jQuery 库 文件 。 


20.2 系统 的 代码 实现 
下 面 分 析 验 证 码 系 统 的 代码 是 如 何 实现 的 。 


20.2.1 系统 主 界面 


在 验证 码 系统 主 界面 中 ， 主 要 显示 验证 码 的 图 片 和 提交 检查 用 户 输入 的 验证 码 ， 如 图 20-2 
所 示 。 
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图 20-2 ”系统 主 界面 
系统 主 界面 的 文件 为 index.php。 主 要 代码 如 下 : 


<?php 
### 验证 码 系统 
### 需要 安装 GD 库 


require 'controller/CodeController.php'; 
session start(); 
if (!lempty($ POST['code'])) { 

$controller = new CodeController(); 

return $controller->checkCode($ POST['code']); 
: 
> 
<!DOCTYPE html> 
<html lang="zh-CN"> 

<head> 


交加 性 07 小 上 
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<meta charset="utf-8"> 
<meta http-equiv="X-UA-Compatible" content="IE=edge"> 
<meta name="viewport" content="width=device-width, initial-scale=1"> 


<title> 验 证 码 系统 </title> 
<link href="css/bootstrap.css" rel="stylesheet"> 


< EE TE gl 
<script 
src="https://cdn.bootcss.com/html5shiv/3.7.3/html5shiv.min.js"></script> 
<script 
src="https://cdn.bootcss.com/respond.js/1.4.2/respond.min.js"></script> 
<ylendifl==> 
</head> 
<body> 
<div class="text-center" style="max-width: 300px; margin: 30px auto;"> 
<h1> 验 证 码 系 统 </h1> 
<br/> 
<br/> 
<div class="form-group"> 
<img id="code-image™" alt=" 验 证 码 ” src="code. php"> 
<a id="refresh" class="btn btn-link btn-block"” href="#"> 换 一 张 </a> 
</div> 
<br/> 
<div class="form-group"> 
<input class="form-control" type="text" name="code" maxlength="4" 
placeholder=" 请 输入 验证 码 ” /> 
</div> 
<div class="form-group"> 
<a id="validate" class="btn btn-primary form-control"> 验 证 </a> 
</div> 
</div> 





<script src="js/jquery-3.2.1.js"></script> 
<script type="text/javascript"> 
$(function() { 
$('#refresh') .on('click', function(e) { 
$('#code-image') .prop('src', 'code.php?t=' + new Date() .getTime() 
Ds; 
$('#validate') .on('click', function(e) { 
var code = $('input [name=code] ') .val (); 


if ($.trim(code) == '') { 
alert (' 请 输入 验证 码 ' ) ; 
return; 


1 
$.post('index.php', {code: code}, function(result) { 
alert (result); 


1D); 
</script> 
</body> 
</html> 


[1 
2 
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DS 
So 
20.2.2 ”生成 并 输入 验证 码 功 能 图 
项 
code.php 文件 的 主要 功能 为 生成 一 个 300x80 的 验证 码 图 片 并 输入 验证 码 。 代 码 如 下 : 员 
<?php 训 
### 生成 验证 码 图 片 i 
### 需要 安装 GD 库 
开 
require 'controller/CodeController.php'; 发 
$controller = new CodeController (); 本 
$controller->generateCode (300, 80); 码 
?> 系 
统 


20.2.3 ”验证 码 控制 器 


文件 CodeController.php 的 主要 作用 是 生成 并 输出 验证 码 图 片 ， 并 验证 提交 的 验证 码 的 正 
确 性 。 主 要 的 方法 如 下 。 

(1) public function checkCode(string $code): 检查 验证 码 是 否 匹配 。 

(2) public function generateCode(int $width = 300, int $height = 80): 生成 随机 验证 码 并 输出 
片 。 

(3) private function saveCode(): 将 验证 码 保 存 到 Session 中 用 于 验证 。 

(4) private function exportImage(int $width, int $height): 生成 并 输出 验证 码 图 片 。 

文件 CodeController.php 的 代码 分 析 如 下 : 

<?php 


### 验证 码 系统 
### 需要 安装 GD 库 


MAG8 


Ee 
* 验证 码 控制 器 
* 生成 验证 码 图 片 并 检查 验证 码 的 正确 性 
class CodeController { 
/** 验证 码 */ 
private $code = ''; 
/** 验证 码 长 度 */ 
private $length = 4; 
/** 验证 码 字符 集 */ 
private $seeds = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXY2Z"'; 


友 太 
* 检查 验证 码 是 否 匹 配 
* @param string $code 验证 码 
ed 
public function checkCode (string $code): bool 
{ 
if (!lempty($code)) { 
if (empty($_SESSION['code'])) { 
echo ' 验 证 码 已 过 期 ， 请 刷新 验证 码 '; 


return false; 
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} 
if (!empty($ SESSION['codeTime'])) { 
$codeTime = (int) $_SESSION['codeTime']; 
$currentTime = time(); 
IE ($currentTime - $codeTime > 180) { 
unset ($_SESSION['code']); 
unset ($_SESSION['codeTime']); 
echo ' 验 证 码 已 过 期 ， 请 刷新 验证 码 ' ; 
return false; 


} 


if (strtoupper($code) != $_SESSION['code']) { 
echo “验证 码 不 匹配 ' ; 


return false; 


} 

# 验证 通过 ， 清 掉 验 证 码 (只 能 验证 一 次 ) 
unset ($_SESSION['code']); 
unset ($_SESSION['codeTime']); 
echo ' 验 证 码 通过 ' ; 


return true; 





| 


return false; 


/** 
* 生成 随机 验证 码 并 输出 验证 码 图 片 
* @param int $width 宽度 (像素 ) 
* @param int $height 长 度 ( 像 素 ) 
eh 
public function generatecode (int S$width = 300, int $height = 80) 
{ 
if ($width <= 0) { 
$width = 300; 
1 
if (Sheight <= 0) { 
Sheight = 80; 
中 


# 生成 验证 码 

Sthis->code = 1 

for ($i = 0, $m = strlen($this->seeds) - 1; $i < $this->length; $i++) 
# 生成 随机 验证 码 
$this->code .= $this->seeds[rand(0, $m)]; 

| 


$this->saveCode () 7 
$this->exportImage ($width, Sheight) 7 


太太 

* 将 验证 码 保存 到 会 话 中 用 于 验证 

入 

w 

private function saveCode() 

{ 
session start(); 
$_SESSION['code'] = $this->code; 
$_SESSION['codeTime'] = time(); 


{ 


session commit(); 


大 六 
* 生成 并 输出 验证 码 图 片 

* @param int $width 宽度 (像素 ) 

* @param int $height 长 度 (像素 ) 

区 

private function exportImage (int $width, int $height) 


{ 
# 生成 验证 码 图 片 


$image = imagecreate($width, S$height); 


# 设置 背景 色 

$backColor = imagecolorallocate ($image, rand(220, 250), rand(220, 
250), rand(220, 250)); 

imagefill ($image, 0, 0, $backColor); 


# 设置 混淆 色 
$maskedColor = imagecolorallocate ($image, rand(180, 220), rand(180, 
220), rand(180, 220)); 
for ($x = 10; $x <= $width; $x += 20) { 
for ($y = 10; $y <= $height; $y += 20) { 
imagefilledellipse ($image, $x, $y, rand(5, 20), rand(5, 20), 
$maskedColor); 


} 
# 设置 字符 色 


$codeColor = imagecolorallocate ($image, rand(150, 200), rand(150, 
200), rand(150, 200)); 

# 加 载 字体 

putenv('GDFONTPATH="' . realpath('.')); 

$font = 'css/font.ttf'; 

# 输出 字体 

$left = $width / ($this->length + 1); 

$top = $height / 2; 

for ($i = 0; $i < $this->length; $i++) { 
imagettftext ($image, S$top, rand(-30, 30), rand(0, $left - 10) + 

Sleft ew (SE 10.5)7 


rand(0, $top) + $top, $codeColor, $font, $this->code[$i]); 


} 
# 输出 验证 码 图 片 


header ('Content-Type: image/png; charset=binary'); 
imagepng ($image); 
imagedestroy ($image); 


20.3 系统 测试 


下 面 测试 验证 码 系统 的 功能 。 有 具体 操作 步骤 如 下 。 


ED 首先 查看 验证 码 系统 的 主页 ， 用 户 可 以 输入 图 片上 的 验证 码 ， 如 图 20-3 所 示 。 
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ECEJRP9 单 击 “验证 ”按钮 ， 弹 出 验证 后 的 信息 ， 如 图 20-4 所 示 。 


























RSS Qe 
验证 码 系统 验证 码 系统 
ie mm 
图 20-3 输入 验证 码 图 20-4 验证 码 通过 





ECDIS 单 击 “ 换 一 张 ” 链 接 ， 即 可 刷新 验证 码 ， 如 图 20-5 所 示 。 
EEC 如 果 输 入 的 验证 码 不 匹配 ， 则 弹出 错误 提示 信息 ， 如 图 20-6 所 示 。 

















| Or 
| 验证 码 系统 

图 20-5 ”刷新 验证 码 图 20-6 ”验证 码 不 匹配 
GEE> 如 果 时 间 超过 3 分 钟 或 者 多 次 输入 验证 码 ， 则 弹出 验证 码 过 期 的 提示 信息 ， 如 


图 20-7 所 示 。 
O Ee 
验证 码 系统 
pe 上 a 
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20-7 ”验证 码 过 期 
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第 21 章 
项 目 实 训 2 一 一 
开发 个 人 博客 


博客 又 称 为 网 络 日 志 ， 是 一 种 通常 由 个 人 管理 、 不 定期 张贴 新 的 文章 的 网 站 。 
本 章 学 习 制 作 个 人 博客 系统 。 在 该 系统 中 ， 用 户 可 以 发 布 、 修 改 和 删除 博客 。 虽 然 
个 人 博客 系统 比较 简单 ， 但 是 读者 可 以 学 习 开 发 动态 网 站 的 流程 、 如 何 使 用 PDO 
操作 MySQL 数据 库 等 知识 。 
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案例 课堂 (第 2 版 ) 8… 





21.1 系统 的 需求 分 析 
在 开发 个 人 博客 系统 之 前 ， 首 先 应 分 析 系统 的 需求 。 
1.1 学 习 目标 
此 验证 码 系统 的 前 台 采 用 Bootstrap 和 jQuery 搭建 ， 后 台 基 于 PHP 7 版 本 ， 使 用 PDO 操 


作 MySQL 数据 库 。 


21 


2 


= 


通过 该 案例 ， 读 者 可 以 实现 以 下 学 习 目 标 。 

(1) 使 用 PDO 连接 MySQL 数据 库 。 

(2) 通过 SQL 语句 执行 数据 表 的 增加 、 修 改 和 删除 操作 。 
(3) 分 别处 理 网 页 的 POST 请 求 和 GET 请 求 。 

(4) 进一步 理解 类 的 定义 和 使 用 方法 。 

(5) 理解 类 的 属性 和 作用 域 。 





.1.2 需求 分 析 


个 人 博客 系统 是 博客 的 生成 与 浏览 管理 系统 ， 该 系统 可 供用 户 记录 学 习 过 程 中 的 知识 


、 疑 难 问题 及 个 人 见解 等 ， 是 个 人 知识 积累 的 一 个 良好 途径 。 


个 人 博客 的 主要 需求 如 下 。 

(1) 用 户 通 过 输入 博客 标题 和 文本 内 容 从 而 生成 一 条 新 的 博客 。 
(2) 用 户 可 以 对 现 有 的 博客 进行 编辑 操作 

(3) 用 户 可 以 对 现 有 的 博客 进行 删除 操作 。 

(4) 在 个 人 博客 首页 中 ， 用 户 可 以 浏览 所 有 已 经 保存 的 博客 。 


.1.3 ”系统 文档 


个 人 博客 系统 的 文档 如 图 21-1 所 示 。 

其 中 核心 文档 的 含义 如 下 。 

(1) index.php: 获取 所 有 博客 并 生成 博客 一 览 页 面 。 

(2) model/BlogModelphp: 博客 数据 模型 ， 使 用 PDO 连接 数据 库 ， 然 后 对 博客 进行 增 
修改 或 删除 等 基本 操作 。 

(3) editphp: 新 建 或 编辑 博客 页 面 ， 在 提交 后 新 建 博客 或 更 新 相应 的 博客 。 
(4) delete php: 根据 ID 删除 相应 的 博客 ， 并 跳 转 到 博客 一 览 页 面 。 

(5) ckeditor 文件 夹 : 在 线 HTML 编辑 器 。 

(6) db.sql: 数据 库 初 始 化 文件 。 

(7) css/bootstrap.css: 设置 页 面 的 样式 。 

(8) js/jquery-3.2.1js: jQuery 库 文件 。 

(9) fonts: 页 面 中 的 字体 文件 。 





« Windows10 (C] ，wamp ，www » code » ch21 


] dbsql 
二 deletephp 

model Beditphp 
PHP 7+MySQLa 坟 网 站 开车 ndexphp 





二 OneDrive 
ET 
9 个 项 目 














21-1 系统 文档 


由 加 各国 ** 四 
和 3 蛙 福 对 生生 上 
:ee 
着 
Se 由 
站 前 
wraa 和 
Et 
: G 
EE 
| 
b S x 
淮 洲 昱 秆 站 潭 着 一 一 Z 兰 半 茵 号 者 z 沁 国 


21.2 数据库 分 析 
分 析 完 系统 的 需求 后 ， 开 始 分 析 数据 库 的 逻辑 结构 并 建立 数据 表 。 


21.2.1 分 析 数 据 库 


本 个 人 博客 系统 的 数据 库 名 称 为 php， 包含 数据 表 blog， 其 结构 如 表 21-1 所 示 。 
表 21-1 blog (博客 数据 表 ) 





[ite | aaes | 但 客 各 和 


| ae | | Nm 


创建 博客 的 时 间 
修改 博客 的 时 间 





21.2.2 ”创建 数据 表 
分 析 数 据 库 的 结构 后 ， 即 可 创建 数据 表 blog。SQL 语句 如 下 : 


CREATE TABLE blog ( 
id int(10) UNSIGNED ZEROFILL NOT NULL AUTO_INCREMENT, 
title varchar (255) NOT NULL, 
content text NOT NULL, 
date created timestamp NOT NULL DEFAULT CURRENT TIMESTAMP, 
date modified timestamp NOT NULL DEFAULT CURRENT_ TIMESTAMP 
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数据 表 创建 完成 后 ， 查 看 效果 ， 如 图 21-2 所 示 。 
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目 洒 和 关 疆 鬼 让 SQL 搜索 缮 插入 辟 导 出 局 导入 oa 权限 .六 操作 法 触发 器 


4 表 结 构 ， 唱 关联 视图 


# 名 字 类 型 排序 规则 属性 空 默认 此 额外 
口 1 idp int(10) UNsGNeDZERorlL 否 无 AUTO_INCREMENT 
口 2 tite varchar(255) utf8_general_ci 和 否 天 
口 3 content text utf8_general_ci 否 无 
口 4 date_created timestamp 否 CURRENT_TIMESTAMP 
口 5 date_modified timestamp 和 否 CURRENT_TIMESTAMP 


21-2 ”数据 表 blog 的 结构 


21.3 个 人 博客 系统 的 代码 实现 
下 面 分 析 个 人 博客 系统 的 代码 是 如 何 实现 的 。 


21.3.1 博客 数据 模型 的 文件 


BlogModel.php 是 博客 数据 模型 文件 ， 主 要 实现 对 数据 库 的 连接 操作 和 对 博客 的 增加 、 删 
修改 等 操作 。 包 含 的 主要 方法 如 下 。 

(1) public function read(): 读 取 所 有 博客 信息 。 

(2) public function find(string $id): 读 取 一 条 博客 信息 。 

(3) public function create(array $params): 创建 一 条 博客 信息 。 

(4) public function update(array $params): 更 新 一 条 博客 信息 。 

(5) public function delete(string $id): 删除 一 条 博客 信息 。 

(6) protected function query(string $sql, array $params = []): 执行 SQL 语句 。 

BlogModel.php 的 代码 如 下 : 

<?php 


### 个 人 博客 系统 
### 需要 安装 PDO 与 PDO_MYSQL 库 


除 


7 


/** 

* 博客 数据 模型 

* 实现 blog 表 的 增删 改 查 
由 


class BlogModel { 
/** 数据 库 连接 串 */ 
Private $dsn = 'mysql:host=localhost;port=3306;dbname=php'; 
/** 用 户 名 */ 


private $user = 'root'; 


/** 密码 */ 

private $password = "123456’ ; 
We 

private $table = 'blog'; 
/** 错误 信息 */ 


private $error 和 


/** 
* 读 取 所 有 博客 

* @return array 
*/ 


public function read(): array 


# 生成 查询 语句 

$sql = "SELECT * FROM {S$this->table}"; 
$result = $this->query ($sql); 

return $result === false ? [] : $result; 


} 


/六 
* 读 取 一 条 博客 

* @param string $id 博客 ID 
* @return array 

这 


public function find(string $id): array 


# 验证 数据 

if (empty($id)) { 
$this->error = 'ID 未 指定 '; 
return []; 


' 
# 生成 查询 语句 


$sql = "SELECT * FROM {S$this->table} WHERE id=:id"; 


$result = $this->query($sql, ['id' => $id]); 
if ($result === false) { 
return []; 
} 
return count($result) > 0 ? $result[0] : $result; 


} 


/#** 
* 添加 一 条 博客 
* @param array $params 博客 内 容 
* @return bool 
public function create (array $params): bool 
加 

# 验证 数据 

if (empty($params['title'])) { 

$this->error =' 标 题 不 能 为 空 '; 


return false; 
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if (empty($params['content'])) { 
$this->error = ' 内 容 不 能 为 空 '; 


return false; 


# 生成 插入 语句 

$sql = "INSERT INTO {$this->table} (title, content, date created, 
date modified) " 
-。 " VALUES (:title, :content, current timestamp, current timestamp)"; 

$result = $this->query($sql, ['title' => $params['title'], 'content' 
=> $params['content']]); 

return $result > 07 





} 


/太太 
* 更 新 一 条 博客 
* @param array $params 博客 内 容 
* @return boolean 
public function update(array $params): bool 
{ 
# 验证 数据 
if (empty($params["'id'])) { 
$this->error = 'ID 未 指定 '; 
return false; 
} 
$fields = []; 
if (!empty($params['title'])) { 
$fields['title'] = $params['title']; 
} 
if (!empty($params['content'])) { 
$fields['content'] = $params['content']; 
1 
if (empty($fields)) { 
$this->error = "请 输入 需要 更 新 的 标题 或 内 容 ' 


return false; 


1 

# 生成 更 新 语句 

$sql = "UPDATE {S$this->table} set "7 

foreach ($fields as $key => $value) { 
$sql .= " {$key}=:{$key}, ™; 

} 

$sql .= " date modified=current timestamp "7 

$sql .= " WHERE id=:id"; 

$fields['id'] = $params['id']; 

$result = $this->query($sql, $fields); 

return $result > 0; 


/** 

* 删除 一 条 博客 

* @param string $id 博客 ID 
* @return bool 


@s 
-| 
第 外 
Ey Sd 
public function delete(string $id) 章 | 
# 验证 数据 项 
if (empty($id)) { 昌 
$this->error = "ID 未 指定 '; 训 
return false; 中 | 
; | 
# 生成 删除 语句 开 
$sql = "DELETE FROM {$this->table} WHERE id=:id"; 秋 
$result = $this->query($sql, ['id' => $id]); 人 
return $result > 0; 博 
| 
A 统 
* 执行 SQL 语句 





* @param string $sql SQL 语句 

* @param array $params SQL 参数 

* @return bool 

ad 

protected function queryl(string $sql, array $params = []) 


{ 

# 连接 数据 库 

$pdo = null; 

try { 
$pdo = new PDO($this->dsn, $this->user, $this->password); 

} catch (PDOException $e) { 
$this->error = ' 数 据 库 连接 错误 : ' . $e->getMessage () 7 
return false; 


AG 





} 


# 执行 SQL 语句 
$stm = $pdo->prepare($sql, [ 
PDO: :ATTR_CURSOR => PDO: :CURSOR_FWDONLY 
1); 
if (!$stm) { 
$this->error = "SQL 语句 或 参数 有 错 ' 7 
} 
if (!$stm->execute($params)) { 
Sthis->error = 'SQL 执行 出 错 : ' . $stm->errorInfo(); 
return false; 


} 
# 获取 返回 结果 


$column = $stm->columnCount (); 
if ($column > 0) { 
# 获取 结果 集 
Srows = $stm->fetchR11 (PDO: :FETCH ASSOC); 
foreach ($rows as &Srow) { 
$row = array_change key case ($row, CASE LOWER); 
} 


return $rows; 
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| 
return $stm->rowCount (); 


} 


/ 

* 获取 错误 信息 

* @return string 

*/ 

public function getError(): string 


{ 





return $this->error; 


用 户 需要 根据 自己 的 数据 库 修 改 $dsn、$user、$password 等 属性 。 





21.3.2 个 人 博客 系统 的 主页 面 


个 人 博客 系统 的 主页 面 为 index.php， 主 要 功能 是 获取 所 有 博客 并 生成 博客 一 览 页 面 。 具 
体 代码 如 下 : 


<?php 
#### 个 人 博客 系统 
### 需要 安装 PDo 与 PDO_MYSQL 库 


require 'model/BlogModel .php'; 


# 获取 所 有 博客 

$model = new BlogModel (); 
$records = $model->read(); 
$error = $model->getError () 
?> 


<!DOCTYPE html> 
<html lang="zh-CN"> 
<head> 
<meta charset="utf-8"> 
<meta http-equiv="X-UA-Compatible" content="IE=edge"> 
<meta name="viewport" content="width=device-width, initial-scale=1"> 


<title> 个 人 博客 系统 </title> 
<link href="css/bootstrap.css" rel="stylesheet"> 


x*I=I1E 1t IE 91> 
<script 
src="https://cdn.bootcss.com/html5shiv/3.7.3/htmlS5shiv.min.js"></script> 
<script 
src="https://cdn.bootcss.com/respond.js/1.4.2/respond.min.js"></script> 


<![endif]--> 
</head> 
<body> 
<div class="container"> 
<div class="row"> 
<div class="col-xs-12"> 
<hl class="text-center"> 个 人 博客 系统 <a class="pull-right glyphicon 
glyphicon-plus" style="text-decoration:none;" href="edit.php"></a></hl> 
<hr/> 
</div> 
</div> 
<div class="row"> 
<div class="col-sm-12 col-md-10 col-md-offset-l1 col-1g-8 col-1g— 
bffset 2"> 
<?php if (!empty($error)) { ?> 
<h3 class="text-center text-danger"><?= $error ?></h3> 
<?php } else if (count($records) == 0) { ?> 
<h3 class="text-center"> 你 还 没有 写 博客 ^^</h3> 
<?php } else { ?> 
<?php foreach ($records as $record) {?> 
<h3> 
<?= $record['title'] ?> 
<span class="text-muted" style="font-size: 70%;">(<?= $record 
['date_created'] ?>)</span> 
<span class="pull-right"> 
<a class="glyphicon glyphicon glyphicon-pencil" style="text-— 
decoration:none;" href="edit.php?id=<?= $record['id'] ?>"></a> 
&nbsp; 
<a class="pull-right glyphicon glyphicon-trash" style="text-— 
decoration:none;" href="#" data-id="<?= S$record['id'] ?>"></a> 
</span> 
</h3> 
Hr > 
<div style="margin: 0 8px; padding: 8px 16px; box-shadow: 0 4px 
4px #eee;"><?= $record['content'] ?></div> 
<br/> 
<br/> 
<?php } ?> 
<?php } ?> 
</div> 
</div> 
<div class="row"> 
<div class="col-xs-12 text-center"> 
<hr/> 
<a class="btn btn-primary btn-lg" style="min-width: 300px; 
="edit .php"> 发 布 博 客 </a> 
<br/> 
<br/> 
<br/> 
</div> 
</div> 
</div> 





href 


i 


将 TU 澡 翰 1z 游 


尝 


车 沙 遇 斑 盖 全 由 局 一 一 2 





MAG8 





PHP+MySQL 动态 网 站 开发 
案例 课堂 (第 2 版 ) 四 一 








<script src="js/jquery-3.2.1.js"></script> 
<script type="text/javascript"> 
$(function() { 
$('.glyphicon-trash') .on('click', function(e) { 
var el = $(this), 
id = el.data('id'); 
if (!confirm(' 你 确定 要 删除 该 条 博客 ? 删除 后 无 法 撤销 ! ')) { 
return; 
$.post('delete.php', { 
id: $(this) .data('id') 
}, function(result) { 
if (result && result.error) { 
alert (result .error); 
return; 





} 
location.reload(); 
}, 'json'); 
Ds; 
]) 7 
</script> 
</body> 
</html> 


21.3.3 个 人 博客 新 建 和 修改 页 面 
editphp 是 新 建 或 修改 博客 页 面 ， 在 提交 后 新 建 博客 或 更 新 相应 的 博客 。 代 码 如 下 : 


<?php 
### 个 人 博客 系统 
### 需要 安装 PDO 与 PDO_MYSQL 库 


require 'model/BlogModel .php'; 
$model = new BlogModel (); 


if (!empty($_POST['blog'])) { 
if (!empty($_POST['blog']l['id'])) { 
# 如 果 POsT 请 求 包 含 博客 及 ID， 更 新 ID 对 应 的 博客 
$model->update($_POST['blog']); 
echo json encode(['error' => $model->getError()]); 
return; 


} 
# 如 果 PosT 请 求 包含 博客 但 是 未 指定 TD， 新 建 一 条 博客 
$model->create($_POST['blog']); 
echo json encode(['error' => $model->getError()]); 
return; 

和 


$record = []; 





$error = "'; 

if (!empty($ REQUEST['id'])) { 
# 如 果 包 含 ID， 获取 ID 对 应 的 博客 用 于 更 新 
Srecord = $model->find($ REQUEST['id']); 
$error = $model->getError (); 


将 TU 易 坤 Lz 游 


D 训 
?> | 
<!DOCTYPE html> 开 
<html lang="zh-CN"> 发 
<head> 上 
<meta charset="utf-8"> 博 
<meta http-equiv="X-UA-Compatible" content="IE=edge"> 客 
<meta name="viewport" content="width=device-width, initial-scale=1"> 系 


<title> 个 人 博客 系统 </title> 





<link href="css/bootstrap.css" rel="stylesheet"> 


5 es 
<script 
src="https://cdn.bootcss.com/html5shiv/3.7.3/html5shiv.min.js"></script> 
<script 
src="https://cdn.bootcss.com/respond.js/1.4.2/respond.min.js"></script> 
<![endif]--> 
</head> 
<body> 
<div class="container"> 
<div class="row"> 
<div class="col-xs-12"> 
<hl class="text-center"> 个 人 博客 系统 </h1> 
<hr/> 
</div> 
</div> 
<div class="row"> 
<div class="col-sm-12 col-md-10 col-md-offset-1 col-1g-8 col-lg- 
offset-2"> 
<?php if (!empty($error)) { ?> 
<h3 class="text-center text-danger"><?= $error ?></h3> 
<?php } else { ?> 
<input type="hidden" name="id" value="<?= $record['id'] ?3? '' ?>" /> 
<div class="form-group"> 
<label for="title"> 标 题 </label> 
<input type="text" name="title" value="<?= $record['title'] ?? 
' 2>" class="form-control" id="title" placeholder=" 标 题 "> 
</div> 
<div class="form-group"> 
<label for="content"> 内 容 </label> 
<textarea id="content" name="content" class="form-control" cols="80" 
placeholder=" 内 容 "><?= $record['content'] ?2? '' ?></textarea> 
</div> 
<br> 
<div class="form-group text-center"> 


AG 
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<button class="btn btn-primary btn-lg" id="submit" type="button™ 
style="min-width: 300px; "> 发 布 博客 </button> 


</div> 


?php } ?> 


<br> 
</div> 
</div> 


<script src="js/jquery-3.2.1.js"></script> 
<script src="ckeditor/ckeditor.js"></script> 


<script type="text/javascript"> 
$(function() { 
CKEDITOR.replace('content'); 


[人 


]) 2 


#submit') .on('click', function(e) { 
Var title = $('#title') .val (); 
if ($.trim(title) == "'') { 
$("'#title') .focus(); 
alert (' 请 输入 标题 ') ; 
return; 
E 


Var content = CKEDITOR.instances .content .getData() 7 


if ($.trim(content) == '') { 
CKEDITOR.instances.content.focus(); 
alert (' 请 输入 内 容 '); 
return; 


1， 
$.post('edit.php', { 
blog: { 
id: $('input[name=id] ') .val(), 
title: title, 
content: content 
} 
}， function(result) { 
if (result && result.error) { 
alert (result .error); 
return; 
} 
location.href = 'index.php'; 
}, "json'); 


</script> 


</body> 
</html> 


21.3.4 个 人 博客 删除 页 面 
delete.php 根据 ID 删除 相应 的 博客 ， 并 跳 转 到 博客 一 览 页面 。 代 码 如 下 : 


<?php 


第 昌 
## 个 人 博客 系统 S| 
### 需要 安装 Ppo 与 PDO_MYSQL 库 章 
require 'model/BlogModel .php'; 到 
$model = new BlogModel (); 浙 


if (!lempty($ REQUEST['id"])) { 
# 根据 ID 删除 对 应 博客 
$model->delete($ REQUEST['id']); 
echo json encode(['error' => $model->getError()]); 
return; 


3 

# 如 果 未 指定 ID， 重 定向 到 首页 
header ('index.php'); 

?2> 


党 洲 昱 种 > 疗 淖 着 一 一 Z 


21.4 系统 测试 


下 面 测试 个 人 博客 系统 的 功能 。 有 具体 操作 步骤 如 下 。 

查看 个 人 博客 系统 的 主页 ， 如 图 21-3 所 示 。 

四 PY 单 击 “ 发 布 博客 ”按钮 或 者 单 击 中 按钮 ， 进 入 创建 博客 页 面 ， 输 入 博客 的 标题 
和 内 容 ， 如 图 21-4 所 示 。 




















© [I voiecshoeiesdo mal od oho 万 -6 OBER py: 
ND We Ee monN I we 
个 人 博客 系统 ] 
可 OO om- oNmm > 状 Gx| 
Er 
向 个 人 博客 系统 + 
as 
人 你 还 没有 写 博 客 ^^ 
SB ST. 
hte , 
Sow 二 或 1009 
图 21-3 个 人 博客 系统 的 主页 图 21-4 编辑 博客 内 容 页 面 


EEC 单 击 “ 发 布 博客 ”按钮 ， 返 回 到 个 人 博客 系统 主页 ， 可 以 看 到 新 添加 的 博客 信 
息 ， 如 图 21-5 所 示 。 

EGG 单 击 编辑 按钮 ， 即 可 编辑 博客 ， 单 击 删 除 按钮 曾 ， 弹 出 警告 信息 ， 单 击 “ 确 
定 ” 按 钮 ， 即 可 删除 博客 ， 如 图 21-6 所 示 。 
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图 21-5 ”查看 新 添加 的 博客 


第 22 章 

项 目 实 训 3 一 一 

开发 用 户 权 限 
系统 


权限 管理 ， 一 般 是 指 根据 系统 设置 的 安全 规则 或 者 安全 策略 ， 用 户 可 以 访问 而 
且 只 能 访问 自己 被 授权 的 资源 ， 不 多 不 少 。 权 限 管理 几乎 出 现在 任何 系统 里 面 。 只 
要 有 用 户 和 密码 的 系统 ， 基 本 都 需要 用 户 权限 管理 。 所 以 本 章 开 始 学 习 用 户 权限 管 
理 系统 的 开发 方法 。 通 过 本 章 的 学 习 ， 读 者 可 以 学 会 如 何 使 用 MVC 模式 来 组 织 代 
码 以 及 MVC 模式 的 优势 ， 使 用 PHP 处 理 具有 一 定 复杂 度 的 多 表 关 系 ， 以 及 如 何 搭 
建 基于 角色 的 用 户 权限 系统 。 
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22.1 必 备 知识 


由 于 该 用 户 权 限 系统 是 采用 MVC 模式 来 组 织 代码 ， 所 以 在 学 习 开发 用 户 管理 系统 之 
前 ， 首 先 需要 了 解 MVC 的 概念 和 原理 。 

MVC 是 Model View Controller 首 字母 的 缩写 。 由 此 可 见 ，MVC 指 的 就 是 “模型 ”“ 哇 
现 ” 和 “控制 器 ”这 3 个 方面 。 其 中 “模型 ”负责 数据 的 组 织 结构 ; “呈现 ”负责 显示 给 浏 
览 者 的 用 户 界面 ， “控制 器 ”负责 业务 流程 逻辑 控制 。 

这 种 结构 的 好 处 有 如 下 几 点 。 

(1) 界面 简单 ， 有 利于 简化 添加 、 删 除 、 修 改 等 操作 。 

(2) 可 以 利用 相同 的 数据 ， 给 出 不 同 的 “呈现 ”。 

(3) 逻辑 控制 的 修改 可 以 变 得 很 简单 。 

(4) 开发 人 员 不 必 重 复 已 经 写 好 的 通用 代码 。 

(5) 有 利于 开发 人 员 共 同 工 作 。 

其 实 ，MVC 结构 是 把 一 个 程序 的 输入 、 处 理 过 程 及 输出 分 开 。 当 用 户 通过 用 户 界 面 输入 
一 个 请 求 的 时 候 ，“ 控 制 器 ” 先 对 请 求 做 出 反应 ， 但 是 “控制 器 ”并 不 真正 地 处 理 数 据 ， 而 
是 调用 “模型 ”和 “呈现 ”中 的 相关 部 分 的 代码 和 数据 来 返回 给 用 户 ， 以 满足 用 户 的 请 求 。 

这 个 过 程 也 可 以 理解 为 : “控制 器 ” 接 到 客户 请 求 ， 以 决定 调用 哪些 “模型 ”中 的 数据 
和 哪些 “呈现 ”方式 。 相 关联 的 “模型 ”通过 相关 业务 规则 处 理 相关 数据 并 且 返 回 。 相 关联 
的 “呈现 ” 则 是 处 理 如 何 格式 化 “模型 ”返回 的 数据 ， 并 且 呈 现 出 最 终结 果 。MVC 结构 的 工 
作 过 程 如 图 22-1 所 示 。 





HTTP 回 应 
HTTP response 


browser 


浏览 器 端 服务 器 端 


browser server 







HTTP request 


22-1 ”MVC 结构 的 工作 过 程 


22.2 系统 的 需求 分 析 


在 开发 用 户 权限 系统 之 前 ， 首 先 应 分 析 系 统 的 需要 。 


22.2.1 学 习 目 标 


此 验证 码 系统 的 前 台 采 用 Bootstrap 和 jQuery 搭建 ， 后 台 基 于 PHP 7 版 本 ， 使 用 PDO 操 
作 MySQL 数据 库 ， 采 用 MVC 模式 来 组 织 代码 。 

通过 该 案例 ， 读 者 可 以 实现 以 下 学 习 目 标 。 

(1) 掌握 如 何 使 用 PDO 连接 MySQL 数据 库 。 

(2) 掌握 如 何 使 用 MVC 模式 来 组 织 代 码 以 及 MVC 模式 的 优势 。 

(3) 掌握 如 何 使 用 PHP 处 理 具 有 一 定 复杂 度 的 多 表 关 系 。 

(4) 进一步 理解 类 的 定义 和 使 用 方法 。 

(5) 理解 类 的 属性 和 作用 域 。 

(6) 掌握 如 何 搭建 基于 角色 的 用 户 权限 系统 。 

该 案例 对 于 读者 学 习 更 为 复杂 的 PHP 框架 具有 很 好 的 启发 和 帮助 作用 。 


22.2.2 需求 分 析 


该 项 目 是 一 个 基于 角色 的 用 户 权 限 系 统 ， 主 要 需求 如 下 。 

(1) 系统 中 有 3 个 模型 : 用 户 (aseD、 角 色 (role) 和 权限 (access)。 

(2) 每 个 用 户 对 应 一 个 角色 ， 角 色 与 角色 之 间 通 过 权限 配置 彼此 的 关系 。 

(3) 基于 不 同 用 户 的 角色 不 同 ， 以 及 角色 之 间 的 权限 配置 ， 一 个 用 户 可 以 对 另 一 个 用 户 拥 
有 不 同 的 权限 。 

(4) 该 项 目 同时 使 用 了 MVC( 模 型 -视图 -控制 器 ) 模 式 ， 该 模式 将 代码 分 为 模型 、 视 图 与 控 
制 器 3 个 部 分 ， 是 一 种 非常 经 典 的 解 耦 模型 。 


























22.2.3 ”系统 文档 
a NS 
用 户 权限 管理 系统 的 文档 如 图 22-2 所 示 。 
| 回国 = 1ch22 - OO x 
主页 3 二 © 
€ 个 Windows10(C) ，wamp > www » code » ch22 ， 。 v 要 22 Pp 
才 快 玉 且 ~ 个 履 日 其 - 
国语 # controller 2017/3/20 1503 
时 全 + css 2017/a/22 1031 
国 阁 + fonts 2017/8/20 1503 
加 时 | i 2017/8/20 1503 
加 model 2017/3/20 1610 
Re view 2017/a/20 1503 
a [dbsaql 2017/5/29 2G21 
PHp 7+MysQLarS a Gindexphp 2017/5/29 2G13 。 PHP 文 
PHP 7 动态 网 站 开发 安 
OneDrve 
a > 
8 个 项 目 选中 1 个 项 目 匡 辐 
22-2 ”系统 文档 
其 中 核心 文档 的 含义 如 下 。 


(1) index.php: 系统 入 口 文 件 ， 调 用 控制 器 实现 页 面 逻辑 关系 。 


将 三 吻 坤 zzZ 涛 


尝 
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(2) view/login.php: 登录 页 面 ， 数 据 库 自 带 一 组 用 户 ， 会 显示 在 该 页 面 中 ， 默 认 密码 是 
123456。 

(3) controllerSiteController php: 页 面 控制 器 ， 主 要 处 理 用户 登 录 和 登 出 ， 以 及 判断 用 户 
的 权限 。 

(4) model/Model.php: 数据 模型 基 类 ， 定 义 数据 库 连 接 和 SQL 执行 。 用 户 需要 根据 自己 
的 数据 库 修改 $Sdsn、$user 和 $password 等 属性 。 

(5) model/UserModel.php: 用 户 数据 模型 ， 处 理 用 户 登 录 和 登 出 ， 以 及 获取 用 户 相关 角色 
和 有 权 访问 的 其 他 用 户 。 

(6) model/RoleModel.php: 角色 数据 模型 ， 获 取 角 色 信 息 以 及 角色 的 相关 权限 (默认 角色 
有 管理 员 、 经 理 、 组 长 和 员工 )。 

(7) model/AccessModel.php: 权限 数据 模型 ， 获 取 权限 信息 ， 并 提供 通过 角色 查找 权限 的 
功能 。 

(8) view/access.php: 权限 显示 页 面 ， 内 含 一 个 可 访问 的 用 户 列表 ， 选 择 不 同 的 用 户 ， 可 
以 查看 对 该 用 户 的 权限 。 

(9) db.sql: 数据 库 初始 化 文件 。 

(10) css/bootstrap.css: 设置 页 面 的 样式 。 

(11) js/jquery-3.2.1.js: jQuery 库 文件 。 

(12) fonts: 页 面 中 的 字体 文件 。 


22.3 数据库 分 析 
分 析 完 系统 的 需求 后 ， 开 始 分 析 数据 库 的 逻辑 结构 并 建立 数据 表 。 


22.3.1 分 析 数 据 库 


用 户 权限 系统 的 数据 库 名 称 为 ppp， 包 含 数据 表 access、role 和 user。 各 个 表 的 结构 分 别 
如 表 22-1 一 表 22-3 所 示 。 


表 22-1 access (权限 数据 表 ) 









号 类 型 
id int(10) 编号 | 主键 
2 本 本 有 int(10) 角色 编号 | 非 空 字段 
3 训 忆 人 int(10) 权限 编号 | 非 空 字段 
| 非 空 字段 
非 空 字段 














4 Tead _ access tinyint(1) 查看 权限 









5 create access tinyint(1) 创建 权限 












update access tinyint(1) 更 新 权限 









delete access 





表 22-2 role( 角 色 数 据 表 ) 











编 号 字段 名 类 型 
和 id int(10) 
2 name varchar(60) 
3 rank int(4) 


编 号 字段 
和 id 

2 name 

3 password 
4 role id 





6 date modi 


表 22-3 user (用 户 数 据 表 ) 


各 
用 户 篇 主人 


用 户 名 称 非 空 字段 
用 户 密码 非 空 字段 
对 应 的 角色 编号 非 空 字段 


5 ee ee 用 户 创建 的 时 间 非 空 字段 
fied 用 户 创建 的 时 间 非 空 字段 





22.3.2 ”创建 数据 表 
分 析 数据 库 的 结构 后 ， 即 可 创造 以 上 3 个 数据 表 。 这 里 采用 导入 数据 库 的 方法 。 


EEIops 打开 


phpMyAdmin， 单 击 “ 导 入 ”链接 ， 如 图 22-3 所 示 。 





eae seal 单 击 界面 中 的 “浏览 ”按钮 ， 如 图 22-4 所 示 。 


i 数据 库 眉 SQL 罗 状态 





中 账户 辐 号 出 A ”更 多 
一 导入 到 当前 服务 器 











铺 修 改 它 到 | | 要 多加 LocalDetabases 
品 服 务 器 活 按 排序 规则 已 (127.0.0.1 via TCPIP) 
pe 回 ”服务 句 关 型: MySQL 要 导入 的 文件 ; 
fOrmbd uricode 可 . ea 
ME Server 总 2 融和 全 于 本 i 
(GPL) 庄 绽 文 编 方式 ] ‘sqlzip 
TT- Re ei 天 | (最 大 限制 : 128 MB) 
。 服务 器 字符 人 : UTF 8 您 可 在 各 页 面 间 拖 放 文件 . 
粤语 言 - Language 硬 : Unicode (uti8) 文件 的 字符 集 : | ud 回 
中 文 - Chinese simplmed 回 | 
22-3 打开 phpMyAdmin 22-4 “导入 到 当前 服务 器 ”界面 


ECDIS 打开 “选择 要 加 载 的 文件 ”对 话 框 ， 选 择 本 章 源 代码 中 的 备份 文件 db.sql， 单 击 


“打开 ”按钮 ， 如 图 22-5 所 示 。 
EEC 单 击 “ 执 行 ” 按 钮 ， 系 统 即 会 读 取 db.sql 文件 中 所 记录 的 指令 与 数据 ， 将 数据 
表 恢 复 ， 如 图 22-6 所 示 。 


全 
下 
yy 


将 开源 需 zz 小 


尝 


汐 沙 尚 范 卫 志 旧书 一 一 人 





AG 
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Et x 
¢ > ~ 小国 ww, code > az» vo [dr F 
ash 要 导入 的 文件 : 
国难 个 名称 四 芋 改 昌明 3 es i 
Ga 办 图 conwcler 2017/a/20 1503 。 > 去 文件 和 必 肌 站 旷 芭 [六 访 式 ] 委 后 如: sqlzip 
me 国电 tie ei 
| php 7 动态 a 四 201718120 1503 。 文 和 4 夫 Cale I 
计 43 纹 才 model 201718120 1610 。 文中 均 文人 的 字符 入 | ul 如 加 
view 201718/20 1503 。 立 中 去 
全 OneDrive abeal 200715/29 2021 SQL 给 档 式 : 
国 tm 证 @ ndexphp 2017/5/29 2013 。 PHP 文件 
| v < > SQL 梧 
EN [dbsal 四 = 
[Lo | ws 
Eee ae | wu 
图 22-5 “选择 要 加 载 的 文件 ”对 话 框 图 22-6 单 击 “ 执 行 ”按钮 
SSTBY 数据 表 创建 完成 后 ， 查 看 access 数据 表 的 结构 ， 效 果 如 图 22-7 所 示 。 
柄 所 accass 
入 搜索 了 插入 已 号 器 导入 5 权限 4 
空 默认 注释 额外 
口 1 dp int(10) meeeopzaaomt 本 大 AUTO_INCREMENT 
口 2 roledp ntto) unslcnEoznoput 否 无 
口 3 access dP int(10) WUL 无 
口 4 read_access linyint(1) 否 0 
口 5 create access tinyint(1) 理 0 
口 5 update_access tinyint(1) 否 0 
口 7 delete_access tinyint(1) 否 0 





图 22-7 数据 表 access 的 结构 
查看 role 数据 表 的 结构 ， 效 果 如 图 22-8 所 示 。 


-| 半 服 全 埠 - Local Databases » ] EE 





关 名 字 类 型 排序 规则 属性 空 默认 注释 额外 
口 1 dp int10) wzeORL 条 无 AUTO_INCREMENT 
口 2 name varchar(60) utf8_general ci 否 无 
口 3 rank int(4) 否 无 


22-8 ”数据 表 role 的 结构 
查看 user 数据 表 的 结构 ， 效 果 如 图 22-9 所 示 。 


售 334 





第 
N 
pcal Databases > : ND 
园 浏 昌 3 结构 让 SQL 对 搜索 怠 插 入 司 导 出 为 导入 aa 权限 吵 操作 到 触发 条 早 | 
3 到 结构 ” 吧 关 联 视 加 项 
且 
着” 名字 类 型 排序 规则 属性 空 默认 对 宏 外 六 
口 1 iudp int(10) Wesszamm 省 才 AUTO_INCREMENT 
口 2 name varchar(60) uti8_general_ci 否 天 | 
口 3 password 。 varchar(50) utf8_general_ci 否 无 开 
as 发 
4 roleid int(10, 2zERopu 否 
mn 成 ii0) 无 用 
口 5 date_created timestamp 否 CURRENT_TIMESTAMP 户 
口 5 date_modified limestamp 否 CURRENT_TIMESTAMP 全 
系 
统 


22-9 数据 表 user 的 结构 





22.4 用 户 权限 系统 的 代码 实现 S 
下 面 分 析 用 户 权限 系统 的 代码 是 如 何 实现 的 。 


22.4.1 系统 入 口 文件 
index.php 为 系统 入 口 文件 ， 调 用 控制 器 实现 页 面 罗 和 辑 关 系 。 代 码 如 下 ; 


<?php 
### 用 户 权限 系统 
### 需要 安装 PDo 与 PDo_MYSQL 库 


A5G 





require 'controller/SiteController.php'; 


# 生成 控制 器 并 响应 请 求 

$controller = new SiteController(); 
$controller->act (); 

De 


22.4.2 页面 控制 器 


SiteController.php 文件 为 页 面 控 制 器 ， 主 要 处 理 用 户 登录 和 登 出 ， 以 及 判断 用 户 的 权限 。 
代码 如 下 : 
<?php 


### 用 户 权限 系统 
### 需要 安装 PDo 与 PDo_MYSQL 库 


require once 'model/UserModel1.php'7 


require once 'model/RoleModel .php'; 
require once 'model/AccessModel .php'; 


和 
5@ 
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/** 


* 控制 器 
* 处 理 逻 辑 并 响应 用 户 请 求 


六 


class SiteController { 


/六 大 


* 分 发 用 户 请 求 


* @return 


ol 


public function act() { 


} 
xx 


$action = $ REQUEST['action'] ?? 'index'; 

$action = 'act' . ucfirst($action); 

if (!method exists($this, $action)) { 
$action = 'actIndex'; 

} 


return $this->$action(); 


* 权限 页 


Sy 


public function actIndex() { 


session_start (); 

# 如 果 用 户 未 登录 ， 跳 转 到 登录 页 

if (empty($_SESSION['loginUser'])) { 
header('Location: index.php?action=login'); 
return; 

} 

$user = $_SESSION['loginUser']; 

$accessUsers = $user->getAccessUsers(); 

if (count ($accessUsers) > 0) { 
# 获取 对 其 他 用 户 的 权限 
$accessID = $_REQUEST['accessID'] ?? $accessUsers[0]->id; 
$access = $user->getAccess ($accessID); 

} 


include 'view/access.php'; 


public function actLogin() { 


session start(); 

if (!empty($_SESSION['loginUser'])) { 
# 如 果 用 户 已 经 登录 ， 跳 转 到 首页 
header ('Location: index.php'); 
return; 


1 


$user = new UserModel (); 

if (!empty($_POST['username']) && !empty($ POST['password'])) 
# 根据 提交 的 用 户 名 和 密码 登录 
if ($user->login($ POST['username'], $_POST['password'])) 


# 用 户 登录 成 功 


{ 


第 
$_SESSION['loginUser'] = $user; SN 
} 章 
echo json encode(['error' => $user->getError()]); 
return; 项 
) 器 
训 
# 取得 所 有 用 户 
$sql = "select u.name, r.name role from user u join role r on u.role_id T 
=r.id"; 开 
$users = $user->query($sql); 发 
include 'view/login.php'; 用 
用 
权 
/太太 了 
* 登 出 统 
六 


public function actLogout () { 
session start(); 
unset ($_SESSION['loginUser']); 
header ('Location: index.php?action=login'); 





J 
Wh 


22.4.3 ”用户 登录 页 面 


login.php 为 登录 页 面 ， 数 据 库 自 带 一 组 用 户 ， 会 显示 在 该 页 面 中 ， 默 认 密 码 是 123456。 
代码 如 下 : 


<!DOCTYPE html> 
<html lang="zh-CN"> 
<head> 
<meta charset="utf-8"> 
<meta http-equiv="X-UA-Compatible" content="IE=edge"> 
<meta name="viewport" content="width=device-width, initial-scale=1"> 


<title> 用 户 权 限 系统 </title> 





<link href="css/bootstrap.css" rel="stylesheet"> 


<I==[iE 1t, TE 91> 
<script 
src="https://cdn.bootcss.com/htmlS5shiv/3.7.3/html5shiv.min.js"></script> 
<script 
src="https://cdn.bootcss.com/respond.js/1.4.2/respond.min.js"></script> 
<!l[endif]--> 
</head> 
<body> 
<div class="container"> 
<div class="row"> 
<div class="col-xs-12"> 
<hl class="text-center"> 用 户 权限 系统 </h1> 
<hr/> 
</div> 
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</div> 
<div class="row"> 
<div class="col-xs-12"> 
<div style="max-width: 320px; margin: 20px auto;"> 
<div class="form-group"> 
<label for="username"> 用 户 名 </label> 
<input type="text" name="username" value="<?= $loginUser->name ?7? 
'' 2>" class="form-control” id="username"” placeholder=" 用 户 名 "> 
</div> 
<div class="form-group"> 
<label for="password"> 密 码 </label> 
<input type="password" name="password" value="" class="form- 
control" id="password" placeholder=" 密 码 "> 
</div> 
<br> 
<div class="form-group text-center"> 
<button class="btn btn-primary btn-lg" id="login" type="button" 
style="min-width: 300px;"> 登 录 </button> 
</div> 
</div> 
</div> 
</div> 
<div class="row"> 
<div class="col-xs-12 text-center"> 
<hr/> 
<h3> 用 户 列 表 (默认 密码 : 123456) </h3> 
<br> 
<div class="text-muted"> 
<?php foreach ($users as $user) { ?> 
<p><?= $user['name'] ?> (<?= $user['role'] ?>)</p> 
< hp 
</div> 
</div> 
</div> 
</div> 


<script src="js/jquery-3.2.1.js"></script> 
<script type="text/javascript"> 
$(function() { 
$('#login') .on('click', function(e) { 
var username = $('#username') .val (); 


if ($.trim(username) == '') { 
$('#username') .focus ()7 
alert (' 请 输入 用 户 名 ' ) ; 
return; 
} 
Var password = $('#password') .val (); 
if ($.trim(password) == '') { 
$('#password') .focus () 7 
alert (' 请 输入 密码 ') ; 
return; 


$.post('index.php', { 
action: "login', 


a 
加 ) 
第 名 
username: USernamer SN 
password: password 章 | 
}, function(result) { 
if (result && result.error) { 项 
alert (result .error); 且 
return; 六 
} 
location.href = 'index.php'; T 
a ys 
ye }, 'json'); 开 
和 发 
DD); 用 
</script> 户 
</body> 权 
</html> 
统 


22.4.4 ”数据 模型 的 文件 
ModeLphp 为 数据 模型 的 文件 ， 定 义 数据 库 连 接 和 执行 SQL 语句 。 代 码 如 下 : 


<?php 
### 用 户 权 限 系 统 
### 需要 安装 PDO 与 PDO_MYSQL 库 





AG 





/*# 
* 数据 模型 
* 实现 数据 模型 的 基础 功能 
让 
class Model { 
/** 数据 库 连接 串 */ 


Private $dsn = 'mysql:host=localhost;port=3306;dbname=php'; 


W239 用 局 各 #7/ 

private $user = 'root'; 
/2** 密码 */ 

private $password = ''; 


/** 错误 信息 */ 
protected $error = "''; 
/** 字段 列表 */ 
protected $fields = []; 


/** 
* 执行 SQL 语句 

* @param string $sql SQL 语句 

* @param array $params SQL 参数 

* @return mixed 

a 

public function query(string $sql, array $params = []) 


{ 
# 连接 数据 库 
$pdo = null; 
ER 
$pdo = new PDO($this->dsn, $this->user, $this->password); 
$pdo->query('set character set utf8°'); 
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} catch (PDOException $e) { 
$this->error = ' 数 据 库 连接 错误 : ' . $e->getMessage(); 
return false; 


. 


# 执行 SQL 语句 
$stm = $pdo->prepare ($sql, [PDO::ATTR CURSOR => PDO: :CURSOR_FWDONLY]) 7 
if (!$stm) { 

$this->error = "SQL 语句 或 参数 有 错 ' ; 


return false; 





if (!$stm->execute($params)) { 
$this->error = 'SQL 执行 出 错 : ' . $stm->errorInfo(); 
return false; 


} 
# 获取 返回 结果 


$column = $stm->columnCount () 7 
if ($column > 0) { 
# 获取 结果 集 
Srows = $stm->fetchAll (PDO: :FETCH RSSOC) 7 
foreach ($rows as &$row) { 
$row = array_change key_case ($row, CASE_ LOWER); 
F 
return $rows; 


} 


return $stm->rowCount () 7 


/太太 

* 填充 模型 数据 

* Q@param array $record 

本 

public function setup (array $record) 

{ 

foreach ($this->fields as $field) { 
if (isset($record[$field])) { 
$this->$field = $record[$field]; 

} 


. 


/*# 
* 获取 错误 信息 

* @return string 

区 

public function getError(): string 
{ 


return $this->error; 


秽 蝇 

第 加 
Se 用 户 需要 根据 自己 的 数据 库 修改 $dsn、$user、S$password 等 属性 。 S 
以 早 
日 
22.4.5 ”用 户 数据 模型 页 面 六 
UserModel.php 为 用 户 数据 模型 页 面 ， 处 理 用 户 登 录 和 登 出 ， 以 及 获取 用 户 相 关 角 色 和 有 T 
权 访 问 的 其 他 用 户 。 代 码 如 下 : 站 
<?php 四 
### 用 户 权限 系统 权 
### 需要 安装 PDo 与 PDo_MYSQL 库 时 
统 


require_ once 'model/Model.php'; 








J/ 
* 用 户 数据 模型 
* 实现 用 户 的 登录 和 角色 查询 
yh 
class UserModel] extends Model { 
XW 
protected S$table = 'user'; 
/Ex 字段 列表 */ 
protected $fields = ['id', 'name', 'role_id', 'date created', 
'date modified', 'role', '‘'accessUsers']; 
/太太 
* 用 户 登 录 
* @param string $username 用 户 名 
* @param string $password 密码 
* @return bool 
入 下 
public function login (string $username, String $password): bool 
{ 
# 验证 数据 


if (empty($username)) { 
$this->error = ' 请 输入 用 户 名 ' ; 
return false; 

} 

if (empty($password)) { 
$this->error = ' 请 输入 密码 '; 


return false; 


| 

# 生成 用 户 验 证 语句 

$sql = "SELECT * FROM {S$this->table} where name=:name and 
password=password(:password) "7 

$result = $this->query($sql, ['name' => $username, 'password' => 
$password]); 

if (empty($this->error) && count ($result) > 0) { 
$this->setup ($result [0]); 
return true; 
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$this->error = "用 户 名 或 密码 有 误 ' 7 


return false; 


/ 
* 获取 用 户 的 角色 
* @return RoleModel 
六 
public function getRole() 
{ 
if (!lisset($this->role)) { 
$this->role = new RoleModel ($this->role id); 





} 


return $this->role; 


/** 
* 获取 用 户 对 其 他 用 户 的 权限 
* @param string $roleID 
* @return unknown 
wf 
public function getAccess(string $userID) { 
$accessUsers = $this->getAccessUsers(); 
foreach ($accessUsers as $accessUser) { 
if ($accessUser->id $userID) { 
$accesses = $this->getRole()->getAccesses(); 
foreach ($accesses as $access) { 
if ($access->access_id == $accessUser->role id) { 
return $access; 





1 
} 


break; 


/** 
* 获取 用 户 有 权 访 问 的 其 他 用 户 
* @return array 
可 
public function getAccessUsers(): array 
{ 
if (!isset($this->accessUsers)) { 
# 获取 用 户 角 色 的 所 有 权限 ID 
$accessIDs = []; 
$accesses = S$this->getRole()->getAccesses (); 
foreach ($accesses as S$access) { 


$accessIDs[] = "'{$access->access_id}'"; 
: 
$accessIDs = '(' . implode(',', $accessIDs) . ')'; 
# 生成 查询 语句 
$this->accessUsers = []; 


$sql = "SELECT * FROM {$this->table} where role id in {$accessIDs} 
order by role id™; 


$result = $this->query($sql1) 7 
if (empty($this->error) && count($result) > 0) { 
foreach ($result as S$record) { 
$model = new UserModel (); 
$model->setup($record); 
$this->accessUsers[] = $model; 


} 
} 


return $this->accessUsers; 


22.4.6 ”角色 数据 模型 页 面 


RoleModel.php 为 角色 数据 模型 页 面 ， 获 取 角 色 信 息 以 及 角色 的 相关 权限 (默认 角色 有 管 


理 员 、 经 理 、 组 长 和 员工 )。 代 码 如 下 : 


<?php 
### 用 户 权 限 系 统 
## 并 需要 安装 PDO 与 PDO_MYSQL 库 


require once 'model/Model .php'; 


class RoleMode1l extends Model { 
Ve 
protected S$table = 'role'; 


X23 字段 列表 


protected $fields = ['id', 'name', 'rank', 'accesses']; 


/太太 
* 构造 函数 
* Q@param string $id 
二 
function _ construct(string $id) { 
if (empty($id)) { 
$this->error = ' 请 提供 角色 ID'; 


return; 


Wt 
# 生成 查询 语句 
$sql = "SELECT * FROM {S$this->table} where id=:id"; 
$result = $this->query($sql, ['id' => $id]); 
if (empty($this->error) && Count ($result) > 0) { 
设置 数据 
Sthis->setup (Sresult[0]) 7: 
return; 
和 
$this->error = ' 角 色 ID 有 误 '; 


yy 


将 鹉 类 需 zz 小 


尝 


汐 沙 河东 了 洒 几 书 一 一 人 





AG 
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/太太 
* 获取 角色 的 权限 
* @return array 
| 
public function getAccesses(): array 
{ 
if (!isset($this->accesses)) { 
# 获取 角色 的 权限 
$accessModel = new AccessModel (); 
$this->accesses = $accessModel->findByRole($this->id); 





} 


return $this->accesses; 


22.4.7 权限 数据 模型 页 面 


AccessModelphp 为 权限 数据 模型 页 面 ， 获 取 权 限 信息 ， 并 提供 通过 角色 查找 权限 的 功 
能 。 代 码 如 下 : 


<?php 
### 用 户 权限 系统 
### 需要 安装 PDo 与 PDO_MYSQL 库 


Freduire_once 'model/Model.php'7 


/* 
* 权限 数据 模型 
a 
class AccessModel extends Model { 
We 
protected S$table = 'access'; 
a 
protected $fields = ['id', 'role_id', 'access_id', 'read access', 
'create access', 'update access', 'delete access']; 
/*# 


* 通过 角色 ID 查询 权限 
* @param string $roleID 角色 ID 
* @return \uc\model\AccessModel[] 
疙 凡 
public function findBYRole (string $roleID): array 
{ 

Sroles = []; 

IE (emptyl($roleID)) { 

$this->error = ' 请 提供 角色 ID'; 


return $roles; 


1 
# 生成 查询 语句 


$sql = "SELECT * FROM {$this->table} where role id=:role id"; 


$result = $this->query($sql, ['role id' => $roleID]); 
if (empty($this->error) && count($result) > 0) { 
foreach ($result as S$record) { 
$model = new AccessModel (); 
$model->setup ($record); 


将 卫 沽 需 zz 小 音 


$roles[] = $model; 训 
} 
| 
return $roles; 开 
发 
? 用 
?> Fr 
权 
有 
22.4.8 权限 显示 页 面 统 





access.php 为 权限 显示 页 面 。 内 含 一 个 可 访问 的 用 户 列表 ， 选 择 不 同 的 用 户 ， 可 以 查看 对 
该 用 户 的 权限 。 代 码 如 下 : 


<!DOCTYPE html> 
<html lang="zh-CN"> 
<head> 
<meta charset="utf-8"> 
<meta http-equiv="X-UA-Compatible" content="IE=edge"> 
<meta name="viewport" content="width=device-width, initial-scale=1"> 


<title> 用 户 权限 系统 </title> 


AG 





<link href="css/bootstrap.css" rel="stylesheet"> 


<—=litf ,1t TE 9 
<script 
src="https://cdn.bootcss.com/html5shiv/3.7.3/html5shiv.min.js"></script> 
<script 
src="https://cdn.bootcss.com/respond.js/1.4.2/respond.min.js"></script> 
nagel = 
</head> 
<body> 
<div class="container"> 
<div class="row"> 
<div class="col-xs-12"> 
<hl class="text-center"> 用 户 权 限 系 统 <a class="pull-right text-primary 
glyphicon glyphicon-log-out" href="index.php?action=logout"></a></hl1l> 
<hr/> 
</div> 
</div> 
<div class="row"> 
<div class="col-xs-12 col-sm-10 col-sm-offset-1 col-md-8 col-md— 
offset-2 col-1g-6 col-lg-offset-3"> 
<div class="form-group"> 
<h3><?= $user->name ?> (<?= $user->getRole()->name ?>)</h3> 
</div> 
<?php if (emptyl($access)) { ?> 
<div class="form-group"> 
<label class="text-danger"> 无 权 查看 其 他 用 户 </1abel> 
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</div> 
<?2php } else { ?> 
<div class="form-group"> 
<labe1> 查 看 用 户 </1abel> 
<select class="form-control" id="access-users"> 
<?2php foreach ($accessUsers as $accessUser) { ?> 
<option value="<?= $accessUser->id ?>" <?= $accessUser->id == 
$accessID ? 'selected' : '' ?>><?= $accessUser->name ?> 
(<?= $accessUser->getRole()->name ?>)</option> 
<?php } ?> 
</select> 
</div> 
<div class="form-group"> 
<p> 查 看 : <span class="glyphicon glyphicon-<?= $access->read access ? 
"ok text-success' : 'remove text-danger' ?>"></span></p> 





</div> 

<div class="form-group"> 

<p> 创 建 : <span class="glyphicon glyphicon-<?= $access->create access ? 
'Ok text-success' : 'remove text-danger' ?>"></span></p> 

</div> 

<div class="form-group"> 

<p> 更 新 : <span class="glyphicon glyphicon-<?= $access->update access ? 
"ok text-success' : 'remove text-danger' ?>"></span></p> 

</div> 

<div class="form-group"> 

<p> 删 除 : <span class="glyphicon glyphicon-<?= $access->delete access 
1ok text-success' : 'remove text-danger' ?>"></span></p> 


要 


</div> 
<?php } ?> 
</div> 
</div> 
</div> 


<script src="js/jquery-3.2.1.js"></script> 
<script type="text/javascript"> 
$(function() { 
$('#access-users') .on('change', function(e) { 
location.href = 'index.php?accessID=' + $(this) .val(); 





Ds; 
Ns 
</script> 
</body> 
</html> 


22.5 系统 测试 


下 面 测 试用 户 权 限 系 统 的 功能 。 具 体操 作 步 又 如 下 。 

EI 查看 用 户 权 限 系统 的 登录 页 面 ， 如 图 22-10 所 示 。 

EEJ2y% 使 用 管理 员 登 录 ， 显 示 管 理 员 可 以 查看 的 用 户 列表 ， 默 认 显示 对 经 理 的 权限 ， 
如 图 22-11 所 示 。 
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图 22-10 ”用户 权限 系统 的 登录 页 面 图 22-11 ”显示 对 经 理 的 权限 


ECDIS 从 “查看 用 户 ” 下 拉 列 表 中 选择 “清风 (组 长 )” 选 项 ， 查 看 管理 员 对 组 长 的 权 
限 ， 如 图 22-12 所 示 。 

EEC 从 “查看 用 户 ” 下 拉 列 表 中 选择 “ 李 四 ( 员 工 )” 选 项 ， 查 看 管理 员 对 员工 的 权 
限 ， 如 图 22-13 所 示 。 
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22-12 ”管理 员 对 组 长 的 权限 22-13 ”管理 员 对 员工 的 权限 


区 TBY 单 击 “退出 ”按钮 [的 ， 返 回 到 用 户 权限 系统 登录 主页 ， 然 后 以 经 理 账号 登录 ， 
默认 显示 对 组 长 的 权限 ， 如 图 22-14 所 示 。 

EECRDp 从 “查看 用 户 ” 下 拉 列 表 中 选择 “ 王 五 (员工 )” 选 项 ， 查 看 经 理 对 员工 的 权限 ， 
如 图 22-15 所 示 。 

ED 单 击 “ 退 出 ”按钮 [区 ， 返 回 到 用 户 权限 系统 登录 主页 ， 然 后 以 组 长 账号 登录 ， 


ua 
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默认 显示 对 员工 的 权限 ， 如 图 22-16 所 示 。 
ED) 单 击 “ 退 出 ”按钮 区 ， 返 回 到 用 户 权 限 系统 登录 主页 ， 然 后 以 员工 账号 登录 ， 
显示 无 权 查看 其 他 用 户 ， 如 图 22-17 所 示 。 
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第 23 章 

项 目 实 训 4 一 一 

开发 社区 市 场 
系统 


社区 市 场 针对 社区 用 户 群 推出 的 物品 分 享 平台 ， 可 以 方便 快捷 地 帮助 社区 用 户 
实现 闲置 物品 的 售卖 或 租赁 ， 从 而 实现 物品 的 高 效 配置 。 为 了 便于 快速 开发 社区 市 
场 系统 ， 使 用 成 熟 的 Yi 框架 搭建 。Yii 是 一 个 高 性 能 的 适用 于 开发 Web 2.0 应 用 
的 PHP 框架， 可 以 大 大 节约 开发 人 员 的 精力 ， 并 对 项 目 品 质 提 供 保障 。 
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23.1 必 备 知识 


由 于 该 社区 市 场 系 统 基于 Yii 2 框架 开发 ， 所 以 读者 需要 了 解 该 框架 的 原理 和 使 用 方法 。 

Yii 是 一 个 基于 组 件 的 高 性 能 PHP 框架 ， 用 于 开发 大 型 Web 应 用 。Yii 采用 严格 的 OOP 
编写 ， 并 有 着 完善 的 库 引 用 以 及 全 面 的 教程 。Yii 提供 了 Web 2.0 应 用 开发 所 需要 的 几乎 一 切 
功能 。 事 实 上 ，Yii 是 最 有 效率 的 PHP 框架 之 一 。 

Yii 是 一 个 高 性 能 的 PHP 的 Web 应 用 程序 开发 框架 。 通 过 一 个 简单 的 命令 行 工 具 yiic 可 
以 快速 创建 一 个 Web 应 用 程序 的 代码 框架 ， 开 发 者 可 以 在 生成 的 代码 框架 基础 上 添加 业务 逻 
辑 ， 以 快速 完成 应 用 程序 的 开发 。 

目前 最 新 的 Yii 框架 为 2.0 版 本 。Yii 2.0 在 PHP 5.4.0 版 本 以 上 完全 重 写 了 Yii， 并 且 完 
全 兼容 PHP 7.x。 它 的 目的 是 成 为 一 个 最 先进 的 新 一 代 的 PHP 开发 框架 。 

在 Wamp 集成 开发 环境 中 ，Yii 2.0 的 安装 是 十 分 简单 的 。 首 先 从 Yii 的 官方 网 站 
http://www.yiiframework.com/ 下 载 Yii 的 最 新 软件 包 ， 如 图 23-1 所 示 。 然 后 将 下 载 的 压缩 包 解 
压 到 Ci\wampWwww\code\ch23 下 ， 然 后 输入 http://localhost/code/ch23/requirements.php， 即 可 
查看 系统 支持 Yii 2.0 框架 的 情况 ， 如 图 23-2 所 示 。 
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23-1 Yii 的 官方 网 站 图 23-2 检查 系统 支持 Yii 框架 的 情况 


23.2 系统 的 需求 分 析 
在 开发 社区 市 场 系统 之 前 ， 首 先 应 分 析 系统 的 需要 。 
23.2.1 学 习 目标 


此 验证 码 系统 的 前 台 采 用 Bootstrap 和 jQuery 搭建 ， 后 台 基 于 PHP 7 版 本 ， 使 用 PDO 操 
作 MySQL 数据 库 ， 使 用 Yi 2.0 框架 开发 社区 市 场 系统 。 

通过 该 案例 ， 读 者 可 以 实现 以 下 学 习 目标 。 

(1) 掌握 如 何 使 用 PDO 连接 MySQL 数据 库 。 
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23.2.2 需求 分 析 


人 
品 


其 他 用 户 提 供 。 


23.2.3 ”系统 文档 


， 可 以 方便 快捷 地 帮助 社区 用 户 实现 闲置 物品 的 售卖 或 租赁 ， 从 而 实现 物品 的 高 效 配置 。 


se—n 
-了 提示 


im 昌 协 8z 游 | 


(2) 掌握 如 何 分 别处 理 POST 请 求 与 GET 请 求 。 

(3) 掌握 如 何 使 用 PHP 处 理 具 有 一 定 复杂 度 的 多 表 关 系 。 

(4) 进一步 理解 类 的 定义 和 使 用 方法 。 

(5) 理解 类 的 属性 和 作用 域 。 

(6) 掌握 如 何 安装 和 使 用 Yi 2.0 框架 。 

该 案例 对 于 读者 学 习 更 为 复杂 的 PHP 框架 具有 很 好 的 启发 和 帮助 作用 。 


党 


该 项 目 是 一 个 基于 Yii 2.0 框架 的 社区 市 场 系 统 ， 主 要 针对 社区 用 户 群 推出 的 物品 分 享 平 


芝 洲 总 填 网 序 驯 首 一 一 


主要 需求 如 下 。 
(1) 系统 有 两 种 不 同 的 用 户 : 社区 用 户 和 社区 代理 。 
(2) 社区 用 户 可 以 在 平台 上 发 布 闲置 物品 供 其 他 用 户 租赁 或 购买 ， 或 发 布 自身 的 需求 以 待 





(3) 社区 用 户 可 以 查看 其 他 用 户 发 布 的 物品 ， 并 根据 需要 租赁 或 购买 。 
(4) 社区 代理 负责 在 用 户 下 单 后 联系 双方 ， 跟 进 交易 状态 ， 以 促进 交易 达成 。 





社区 市 场 系 统 的 文档 如 图 23-3 所 示 。 
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23-3 ”系统 文档 结构 


该 项 目 由 Yii 2 的 基本 应 用 程序 模板 开发 而 来 ， 目 录 结 构 与 模板 保持 一 致 ， 主 要 

人 。 有 config、controllers、models、views、web 和 vendor 等 。 其 中 vendor 目录 下 主要 是 
Yii 2 的 框架 和 依赖 库 ， 不 需要 修改 ， 其 他 目录 Yii 2 提供 了 一 个 简单 的 基础 页 面 结 
构 ， 该 项 目的 业务 文件 基本 都 是 新 加 的 。 

其 中 核心 文档 的 含义 如 下 。 

(1) config/web.php、db.php、params.php: 项 目 配置 文件 ， 用 户 需要 修改 db.php 中 的 数据 


. 
1@ 
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库 配 置 以 连接 自己 的 数据 库 。 

(2) web/index.php: 系统 入 口 文件 ， 所 有 的 请 求 都 通过 该 文件 响应 。 

(3) web/ 其 他 文件 : 包括 网 页 的 样式 表 、 脚 本 和 字体 等 前 端 资源 文件 。 

(4) controllers/TtemControllerphp: 物品 与 需求 展示 控制 器 ， 主 要 实现 物品 与 需求 的 展示 、 
搜索 、 下 单 以 及 物品 内 留言 的 功能 ， 同 时 处 理 用 户 的 登录 、 登 出 和 注册 。 

(5) controllers/AdminController.php: 用 户 后 台 控 制 器 ， 主 要 帮助 用 户 发 布 与 管理 物品 和 需 
求 ， 以 及 查看 订单 。 对 于 代理 用 户 ， 主 要 用 于 更 新 交易 状态 。 

(6) models/*.php: 数据 模型 文件 ， 主 要 实现 数据 表 category( 物 品 分 类 )、comment( 留 言 )、 
customer( 用 户 ) 、item( 物 品 ) 、item_img( 物 品 图 片 )、need( 需 求 ) 、need_img( 需 求 图 片 )、 
timetype( 时 间 类 型 )、transaction( 交 易 ) 和 type( 交 易 方 式 ) 的 模型 。 

(7) views/*.php: 视图 文件 ， 主 要 实现 各 个 页 面 的 html 内 容 。 

(8) views/item/*.php: 项 目 新 加 文件 ，ItemController 控制 器 对 应 的 视图 文件 ， 包 括 该 控制 
器 下 的 各 个 页 面 的 具体 实现 。 

(9) asserts/*.php: 前 端 资源 包 管理 文件 ， 整 合 相 应 的 前 端 资源 供 页 面 使 用 。 

(10) db.sql: 数据 库 初 始 化 文件 。 默 认 社 区 代理 账号 是 : agent@test.com ， 密 码 是 
123456。 

(11) widget/*.php: 页 面 小 部 件 ， 生 成 定制 按钮 。 

(12) runtime/logs/ app.log: 系统 运行 日 志文 件 。 





23.3 还原 数据 库 


社区 市 场 系统 的 数据 库 名 称 为 trade， 包 含 10 数据 表 : category、comment、customer、 
item、item_img、need、need img、 timetype、 transaction 和 type。 

使 用 phpMyAdmin 工具 导入 本 章 的 数据 库 备 份 文件 db.sql， 即 可 将 数据 表 恢 复 ， 结 果 如 
图 23-4 所 示 。 





a 昌 导 k 局 导入 六 操作 2 权限 壤 程 序 加 事件 更 多 





Mm 
表 。 操作 行政 入 类 型 排序 规则 大 小 多余 
口 oategory ”请 同 训 览 六 结构 号 搜索 了 插入 志清 空 @ 删 除 10 InnoDB utf8_general di 16 KB 


口 comment 富 丫 浏 览 jr 结构 志 扫 索 腾 插 入 铝 清 主 日 别 除 1 InnoDB uti8 general cd 16 KB 


口 customer 请 蛋 浏 览 级 结构 他 搜索 落 插 入 赐 清 空 @ 惑 除 3 InnoDB utf8 general cl 48 FB 
四 一 请 目 浏览 关 车 鬼 名 搜索 匡 插 入 希 清 二 目 帅 除 1 InnoDB utis general di 16 KB 
口 em_img 请 国 浏览 纠结 构 他 搜索 了 括 入 锋 清 宇 @ 删除 1 InnoDB uti8 general cl 16 KB 
口 "eed 寅 日 浏 生 六 结 鬼 志 接 索 匡 插 入 急 清 宇 日 弄 除 1 InnoDB utf8 general ai 16 KB 
口 "eed_img 仿 园 浏 览 纹 结 鬼 他 搂 末 卫 插 入 多 清 宇 日 型 除 0 InnoDB latin1_swedish_ci 16 KB 
口 timetype ”请 司 训 览 大 千 鬼 包 搜 索 甘 插 入 得 清 宇 日 开除 5 InnoDB latint_swedish ci 16 KB 
口 transact tion 人 富 局 浏 览 级 结构 让 搜索 也 插入 志清 宇 中 开除 0 InnoDB uti8 general ci 16 KB 
口 gpe 富 目 浏 览 六 结 鬼 包 接 索 甘 插 入 同 清 室 日 得 除 3 InnoDB latinl_swedish ci 16 FB 
10 张 表 总 计 26 MylSAM utf8_general_ci 192 KB 0 字 节 


23-4 还 原 数据 表 


第 

23.4 ”社区 市 场 系统 的 代码 实现 | 

目 

下 面 分 析 社 区 市 场 系统 的 代码 是 如 何 实现 的 。 六 


23.4.1 系统 入 口 文 件 
index.php 为 系统 入 口 文件 ， 所 有 的 请 求 都 通过 该 文件 响应 。 代 码 如 下 : 


<?php 


// defined('YII DEBUG') or define('YIIT DEBUG', true); 
// defined('YII ENV') or define('YII ENV', 'dev'); 


车 油 东 寸 风 兰 由 书 一 一 光 





require(_DIR  . '/../vendor/autoload.php'); 
require(_DIR  . '/../vendor/yiisoft/yii2/Yii.php'); 


$config = require(_DIR  . '/../config/web.php'); 


(new yii\web\Application($config))->run(); 


23.4.2 ”项 目 配 置 文 件 


项 目 源 代码 中 的 config 文件 夹 下 包含 了 3 个 配置 文件 ， 即 db.php、web.php 和 params.php 
文件 。 
db.php 文件 配置 了 连接 数据 库 的 参数 。 代 码 如 下 : 


<?php 





return [ 
'class' => 'yii\db\Connection', 
"dsn' => 'mysql:host=localhost;dbname=trade', 
'UuUsername' => 'root', 
'password' => '123456°', 
'charset' => 'utf8', 
让 
2> 


params.php 文件 自行 定义 的 一 些 全 局 信息 ， 主 要 是 物品 分 类 、 购 买方 式 等 。 代 码 如 下 : 
<?php 
return [ 

"adminEmail' => 'admin@yourhost.com', 


"categories' => [ 


' 全 部 '， 


大 
53@ 
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"音乐 '， 
' 玩 具 '， 
' 衣 物 '， 
' 家 居 '， 
' 其 他 ' 

]， 

'itemTypes' => [ 
"rent' => "出 租 '， 
rsale， =>' 出 售 '， 

], 

"needTypes' => [ 
'rent' => ' 租 借 '， 
"sale' =>' 购 买 '， 
'rent_or_sale' => ' 租 借 或 购买 '， 





]， 

"transactionStatuses' => [ 
'requested' => ' 已 下 单 '， 
'in-rent' => ' 租 赁 中 '， 
'complete' =>' 结 束 '， 

], 

'RentPeriods' => [ 

SR 
' 周 '， 
ale 
' 季 度 '， 
“年 '， 
]， 
] 
Te 


web.php 为 项 目 主要 配置 文件 ， 大 多 数 可 以 采用 默认 配置 ， 项 目 主 要 修改 了 
defaultRoute( 默 认 控 制 器 ItemControlleD 和 layout( 默 认 模 板 admin.php)。 代 码 如 下 : 


<?php 
$params = require (_DIR  . '/params.php'); 


$config [ 
"id' => 'basic', 
'basePath' => qirname (_DIR_)， 
'bootstrap' => [ 
rlog! 
]， 
"components' => [ 
"request' => [ 
// !!! insert a secret key in the following (if it is empty) 一 
this is required by cookie validation 
"cookieValidationKey' => 'rentValidation’' 
]， 
cache" => [ 
'class' => 'yii\caching\FileCache' 


"user' => [ 
"identityClass' => 'app\models\User', 


G5 


"enableRAutoLogin' => true 
]， 
Ea 
'errorHandler' => [ 
"errorRction' => "item/error' 
]， 
/* 
"mailer' => [ 
"class'! => 'yii\swiftmailer\Mailer', 
// send all mails to a file by default. You have to set 


// 'useFileTransport' to false and configure a transport 


// for the mailer to send real emails. 
'useFileTransport' => true 


'traceLevel' => YII_DEBUG ? 3 : 0, 
"targets' => [ 
[ 
"class' => 'yii\log\FileTarget', 
'levels' => [ 
TY 
"warning' 


]， 

"db'， => require (_DIR__ . '/db.php'), 

/* 

"urlManager' => [ 
"enablePrettyUr1' => true, 
"showScriptName' => false, 
"rules' => [] 

] 

区 

]， 

'language' => 'zh-CN', 
'defaultRoute' => 'item', 
'layout' => 'admin', 
'params' => $params, 


(YII_DEBUG) { 
// configuration adjustments for 'dev' environment 
$config ['bootstrap'] [] = 'debug'; 
$config ['modules'] ['debug'] = [ 
'class' => 'yii\debug\Module', 


'allowedIPs' => [ 
1 


]; 


sconfig [bootatrap] [] = "gii"s 
$config ['modules'] ['gii'] = [ 


涯 将 澡 需 8c 小 音 


淮 洲 总 填 网 序 掉 着 一 一 
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"class' => 'yii\gii\Module!' 


return $config; 
EE 


23.4.3 ”物品 与 需求 展示 控制 器 


ItemControllerphp 为 物品 与 需求 展示 控制 器 ， 主 要 实现 物品 与 需求 的 展示 、 搜 索 、 下 单 
以 及 物品 内 留言 的 功能 ， 同 时 处 理 用 户 的 登录 、 登 出 和 注册 。 代 码 如 下 : 


<?php 
namespace app\controllers; 


use app\models\Customer, app\models\Item, app\models\Need, 

app\models\ItemImg, app\models\NeedImg, app\models\TimeType, 
app\models\Category, app\models\Type, app\models\Comment, 

app\models\Transaction; 

Use Yiiy 

use yii\data\Pagination; 


J/ 
* 物品 与 需求 展示 控制 器 
wy 
class ItemController extends \yii\web\Controller { 


/太太 


public $layout = 'item'; 


/太太 
* @inheritdoc 
s 
public function actions () 
{ 
return [ 
"error' => [ 
"class' => 'yii\web\ErrorAction', 
]， 


/** 

* 首页 

public function actionIndex() { 
Sthis->layout = 'index'; 


// 取得 最 新 物品 
S$whereConditions = "status = 'available' and display = 1"7 
$model = $items = Item: :find()->where ($whereConditions)— 


>orderBy('priority desc, id desc')->limit(16)->all(); 


} 


$products = []; 
if (!empty($model)) { 
foreach ($model as $item) { 


// 取得 物品 图 片 


mg 十 cz 关 国 





$itemImages = ItemImg: :find()->where([ 训 

"item_ id' => $item['id'] | 
])->asArray () ->all (); 

$images = []; 国 

if (!empty($itemImages)) { 社 

foreach ($itemImages as $image) { 区 

$images[] = $image['images']; 市 

! 有 

} 统 


$temp = $item->attributes; 

$temp['images'] = $images; 

// 取得 租赁 物品 时 间 类 型 

$timeType = TimeType::findOone($temp['timetype_id']); 
$temp['timetype'] = $timeType["name"]; 

$products[] = $temp; 





AG 





} 
// 泻 染 页 面 


return $Sthis->render('home'，[ 
"products' => $products 


]) 7 


J 
* 需求 一 览 页 
* @param string $redirect 


public function actionNeed($redirect = false) 


{ 


desc 





if (!strstr(Yii::$app->request->referrer, 'need') && S$redirect 
false) { 
$this->clearRelatedSession(); 


} 
// 取得 符合 条 件 的 所 有 需求 


$whereConditions = "status = 'available' and display = 1"7 
if (\Yii::$app->session->has('Conditions')) { 
// 种 类 


$whereConditions .= ' and ' . \Yii::$app->session->get('Conditions'); 


if (\Yii::$app->request->get('search', false)) { 
// 搜索 框 内 容 
$search = \Yii::$app->request->get ('search'); 
$whereConditions .= ' and name like \'%' . $search . '%\''; 
1 


$needs = Need: :find()->where ($whereConditions)->orderBY('PIiority 
+ id desc')s 
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if (\Yii::$app->request->get('category', false)) { 
// 设置 种 类 
$category = \Yii::$app->request->get ('category'); 
switch ($category) { 
case 1 : 
\Yii 
break; 
case 2 
\Yii 
break; 
case 3 : 
\Yii 
break; 
case 4 
\Yii 
break; 
case 5 





$app->session->set ('Toolsstatus', 1); 








$app->session->set ('SportsGoodsStatus', 1); 





$app->session->set ('Booksstatus', 1); 








$app->session->set ('VideoGamesStatus', 1); 


app->session->set('MusicInstrumentsStatus'，1)7 


$app->session->set ('ToysAdGamesStatus', 1); 


case 7 
app->session->set('ClothStatus'，1)7 











$app->session->set ('Householdstatus', 1); 
break; 

case 9 : 
Mri 
break; 





$app->session->set ('OthersStatus', 1); 


| 
$category_str = 'category id = ' . $category; 
\Yii::$app->session->set('Conditions', $category_str); 
Sthis->redirect([ 

'item/need', 

'bool' =>1 





// 设置 翻 页 
$pages = new Pagination([ 
"totalCount' => $needs->count(), 
'pageSize' => '6' 
YY 
$model = $needs->offset ($pages->offset)—->limit ($pages->1limit)— 
>with('type')->with('timetype')->all(); 


$wanteds = []; 
if (!empty($model)) { 
foreach ($model as Sneeds) { 


// 取得 需求 图 片 


SneedImages = NeedImg::find()->where([ 


need id' => $needs['id'] 
1)->asArray()->all(); 
$images = []; 
if (!empty($needImages)) { 
foreach ($needImages as $image) { 
$images[] = $image['images']; 
,i 
} 
$temp = $needs->attributes; 
$temp['images'] = $images; 
$temp['type'] = $needs->type; 
$temp['timetype'] = $needs->timetype; 
$wanteds[] = $temp; 


1 
// 演 染 页 面 


return $this->render('need', [{ 
'wanteds' => $wanteds, 
'pages' => $pages 

]) 7 


淮 济 剖 填 网 记 淖 半 一 一 兰 将 耿 昌 性 8Z 滤 重 


} 


/太太 
* 物品 一 览 页 

* @param string $redirect 
pe 


public function actionItem($redirect = false) 
{ 
if (!strstr(Yii::$app->request->referrer, '‘'item') && S$redirect == 
false) { 
$this->clearRelatedSession(); 


} 
// 取得 符合 条 件 的 所 有 物品 


$whereConditions = "status = 'available' and display = 1"; 
if (\Yii::$app->session->has('Conditions')) { 


// 种 类 


$whereConditions .= ' and ' . \Yii::$app->session->get('Conditions'); 
if (\Yii::$app->session->has('Pricestr', false)) { 
// 价格 


S$whereConditions .= ' and ' . \Yii::$app->session->get('Pricestr'); 


if (\Yii::$app->request->get('search', false)) { 


// 搜索 框 内 容 
$search = \Yii::$app->request->get ('search'); 
$whereConditions .= ' and name like \'%' . $search . '%\''; 


} 
$items = Item: :find()->where ($whereConditions)—->orderBy('priority 
desc, id desc'); 


if (\Yii::$app->request->get('category', false)) { 


// 设置 种 类 
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$category = \Yii::S$Sapp->request->get('category')7 
switch ($category) { 








$app->session->set ('Toolsstatus', 1); 


break; 
case 2 : 





WW 
break; 
case 3 
Wh 
break; 
case 4 : 
\Yii 
break; 
case 5 
\Yii 
break; 
case 6 


$app->session->set ('SportsGoodsStatus', 1); 








$app->session->set ('Booksstatus’', 1); 





$app->session->set ('VideoGamesStatus', 1); 








$app->session->set ('MusicInstrumentsSstatus', 1); 


app->session->set ('ToysAdGamesStatus', 1); 


$app->session->set('Clothstatus', 1); 


case 8 
app->session->set ('Householdstatus', 1); 











case 9 
\Yii::$app->session->set ('Othersstatus’', 1); 
break; 
} 
$category_str = 'category id = ' . $category; 


\Yii::$app->session->set('Conditions', $category_str); 
$this->redirect([ 

'item/item', 

'bool' =>1 
1); 


// 设置 翻 页 
$pages = new Pagination([ 
"totalCount' => $items->count(), 
'pageSize' => '!15" 
1); 
$model = $items->offset ($pages->offset)->limit ($pages-—>limit)->all(); 


$products = []; 
if (!empty($model)) { 
// 取得 物品 图 片 
foreach (Smodel as $item) { 
$itemImages = ItemImg::find()->where([ 
"item id' => $item['id'] 
])->asArray ()->all (); 
$images = []; 
if (!empty($itemImages)) { 





第 
foreach ($itemImages as $image) { S| 
$images[] = $image['images']; 章 
} 
9 项 
$temp = $item->attributes; 目 
$temp['images'] = $images; 六 
$products[] = $temp; 
] 
。 开 
发 
// 泻 染 页 面 往 
return $this->render('item', [{ 区 
"products' => $products, 市 
"pages' => $pages 要 
1); 统 
} 
/** 
* 检索 物品 (AJAX， 按 价格 区 间 ) 
*/ 


public function actionItemPriceRAjax() 


{ 





if (!\Yii::$app->request->isAjax) { 
$url = str_replace('item-price-ajax', 'item', \Yii::$app->request-— 
>url); 
return $this->redirect([ 
$url 
De 





$this->layout = false; 


$conditions = \Yii::$app->session->get('Conditions', false); 
$price = \Yii::$app->request->post('Price'); 
$search = \Yii::$app->request->post('search', false); 


// 价格 区 间 检 索 条 件 
$price_str = false; 
switch ($price) { 
case “1 3 
$price_str = " ((rent price > 0 and rent price < 11) OR 
(sale price > 0 and sale price < 11))"; 
\Yii::$app->session->set ('Price', 1); 
break; 
Cae 
$price_ str = " ((rent price > 10 and rent price < 31) OR 
(sale price > 10 and sale price < 31))"; 
\Yii::$app->session->set('Price', 2); 
break; 
cane 3 3 
$price str = " ((rent price > 30 and rent price < 51) OR 
(sale price > 30 and sale price < 51))"; 
\Yii::$app->session->set ('Price', 3); 
break; 
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Case "4 : 
$price_ str = " ((rent price > 50 and rent price < 101) OR 
(sale price > 50 and sale price < 101))"; 
\Yii::$app->session->set ('Price', 4); 
break; 
Case "5" : 
$price str = " ((rent price > 100 and rent price < 201) OR 
(sale price > 100 and sale price < 201))"; 
\Yii::$app->session->set('Price', 5); 
break; 
Case "6' : 
$price_str = " (rent price > 200 OR sale price > 200)"; 











\Yii::$app->session->set ('Price', 6); 
break; 
\ default : 

N \Yii::$app->session->remove('Price'); 
\Yii::$app->session->remove('Pricestr'); 
break; 

} 
// 检索 符合 条 件 的 所 有 物品 
$whereConditions = "status = 'available' and display = 1"7 
if ($price_str) { 
// 价格 
\Yii::$app->session->set ('Pricestr', S$price_str); 
$whereConditions .= ' and ' . S$price str; 
i 
if ($conditions) { 
// 种 类 
$whereConditions .= ' and ' . $conditions; 
中 
if ($search) { 
// 物品 名 
$whereConditions .= ' and name like \'%' . $search . '%\''; 
J 


$items = Item: :find()->where ($whereConditions)->orderBy('priority 
desc, id desc'); 


// 设置 分 页 
$pages = new Pagination([ 
"totalCount' => $items->count(), 
"pageSize' => '9' 
二 
S$model = $items->offset($pages->offset)->limit($pages-—>limit)->all (); 


Sproducts = []7 
if (!empty($model)) { 
foreach ($model] as $item) { 
// 获得 物品 相关 图 片 
$itemImages = ItemImg::find()->where([ 
"item id' => $item['id'] 

])->asArray()->all (); 
$images = []; 
if (!empty($itemImages)) { 


foreach ($itemImages as $image) 1{ 
$images[] = $image['images']; 
上 
} 
$temp = $item->attributes; 
$temp['images'] = $images; 
$products[] = $temp; 


将 Tm 澡 则 8z 游 全 


尝 


1 
// 泻 染 页 面 


return $this->render('item-ajax', [ 
"products' => $products, 
'pages' => $pages 


警 沙 东 寸 风 兰 由 书 一 一 


])7 
} 





/** 

* 检索 物品 (AJAX) 

二 
public function actionItemAjax() 
{ 





if (!\Yii::$app->request->isAjax) { 
$url = str_replace('list-ajax', 'list', \Yii::$app->request->url); 
return $this->redirect([ 
$url 





bh 
$this->layout = false; 
$category \Yii::$app->request->post('Category', false); 


$search = \Yii::$app->request->post('search', false); 
$price_str = \Yii::$app->session->get ('Pricestr', false); 





Sthis->recordCheckBoxStatus (); 


// 取得 所 有 符合 条 件 的 物品 


$whereConditions = "status = 'available' and display = 1"7 
if ($price_str) { 
// 价格 
$whereConditions .= ' and ' . S$price_ str; 
} 
if ($category) { 
// 种 类 
$conditions = 'category id in ' . $category; 
\Yii::$app->session->set('Conditions', $conditions); 
$whereConditions .= ' and ' . $conditions; 
} else { 
\Yii::$app->session->remove('Conditions'); 
} 
if ($search) { 
// 物品 名 
$whereConditions .= ' and name like \'%' - $search - '%\''; 
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| 


$items = Item: :find()->where ($whereConditions)->orderBY('Priority 
desc, id desc'); 


// 设置 翻 页 
$pages = new Pagination([ 
"totalCount' => $items->count(), 
"pageSize' => "15" 
Ji 
$model = $items->offset ($pages->offset)->1imit(S$pages->1imit)->al1()7 





$products = []; 
if (!lempty($model)) { 
// 取得 物品 图 片 
foreach ($model as $item) { 
$itemImages = ItemImg::find()->where([ 
"item id' => $item['id'] 
])->asArray()->all(); 
$images = []; 
if (!empty($itemImages)) { 
foreach ($itemImages as $image) { 
$images[] = $image['images']; 
» 
} 
$temp = $item->attributes; 
$temp['images'] = $images; 
$products[] = $temp; 


} 
// 演 染 页 面 


return $this->render('item-ajax', [ 
"products' => $products, 
'pages' => $pages 
1); 
} 


/* 大 
* 检索 需求 (AJAX) 
= 
public function actionNeedAjax() 
{ 
if (!\Yii::$app->request->isAjax) { 
$url = str_ replace('need-ajax', 'need', \Yii::$app->request->url); 
return Sthis->redirect([ 
$url 





} 
$this->layout = false; 


$category = \Yii::$app->request->post('Category', false); 
$search = \Yii::$app->request->post('search', false); 


| 
第 /ree 
$this->recordCheckBoxStatus () 7 SI 
// 检索 符合 条 件 的 所 有 需求 加 
S$whereConditions = "status = "available' and display = 1"; 项 
if ($category) { 目 
// 种 类 济 
$conditions = 'category id in ' . $category; 
\Yii::$app->session->set('Conditions', $conditions); | 
$whereConditions -= ' and ' . $conditions; 
Velse 二 站 
\Yii::$app->session->remove('Conditions'); 社 
1 区 
if ($search) { 市 
// 物品 名 
$whereConditions .= ' and name like \'%' . $search . '%\''; 统 


$needs = Need: :find()->where ($whereConditions)->orderBy('priority 
desc, id desc'); 





// 设置 翻 页 

$pages = new Pagination([ 
"totalCount' => $needs->count(), 
"pageSize' => '6' 


AG 





1); 
$model = $needs->offset ($pages->offset)->limit ($pages->limit)— 
>with('type')->with('timetype')->all(); 


$wanteds = []; 
if (!lempty($model)) { 
foreach ($model as $needs) { 
// 获取 需求 图 片 
$needImages = NeedImg::find()->where([ 
'need_id' => $needs['id'] 
])->asArray () ->all (); 
$images = []; 
if (!lempty($needImages)) { 
foreach ($needImages as $image) { 
$images[] = $image['images']; 
ys 
} 
$temp = $needs->attributes; 
$temp['images'] = $images; 
$temp['type'] = $needs->type; 
$temp['timetype'] = $needs->timetype; 
$wanteds[] = $temp; 


} 
// 演 染 页 面 


return $this->render('need-ajax', [ 
'wanteds' => S$wanteds, 
'pages' => $pages 

I 
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/*# 
* 物品 详细 页 

* @param string $id 物品 ID 

wy 

public function actionIitemDetail ($id) 


{ 

// 取得 物品 与 相关 图 片 

$model = Item: :findone ($id); 

$itemImg = ItemImg::find()->where([ 
"item id' => $id 

] ) ->asRrray()->al1()7 

// 取得 留言 

$itemComments = Comment::find()->where([ 
"item id' => $id 

])->orderBy('create at desc')->all(); 





// 泻 染 页 面 
return $this->render('single', [ 
'model' => $model, 
'images' => $itemImg, 
"comments' => $itemComments 
1); 
} 


/太太 
* 需求 详细 页 

* @param string $id 需求 ID 

SE 

public function actionNeedDetail ($id) 


{ 
// 获取 需求 与 相关 图 片 
$model = Need::findone ($id); 
$needImg = NeedImg::find()->where([ 
'need id' => $id 
])->asArray()->all(); 


return $this->render('wanted-single', [ 
model' => $model, 
'images' => $needImg 


} 


/** 
* 发 布 留言 
wh 
public function actionCommentAjax() 
{ 
if (\Yii::$app->request->isAjax) { 
Sthis->layout = false; 


// 取得 请 求 数据 


$itemId = \Yii::$app->request->post('ItemId'); 


[ 
国 加 
Sl 


$customerId = \Yii::$app->request->post('CustomerId'); 
$message = \Yii::$app->request->post('Comment'); 


// 新 建 留言 

$comment = new Comment () 7 
$comment->item id = $itemId; 
$comment->customer id = $customerId; 
$comment->comment = $message; 
$comment->create at = time(); 


将 卫 沽 需 8z 小 


尝 


if ($comment->save()) { 
// 取得 所 有 留言 
$model = Comment::find()->where([ 
"item id' => $itemId 
])->orderBy('create at desc')->all(); 
return $this->render('comment-ajax', [ 
'model' => $model 


淮 洲 总 填 网 序 掉 着 一 一 





]); 
} else { 
$message = ' 留 言 失败 ， 请 稍 候 重 试 '; 
return json encode([ 
'error_code' => 2001, 
"error message' => $message 


AG 





public function actionRequest () 
. 
if (\Yii::$app->request->isAjax) { 
$clientId = \Yii::$app->request->post ('ClientId', false); 
$itemId = \Yii::$app->request->post('ItemId', false); 
SneedTime = \Yii::$app->request->post ('NeedTime', false); 


// 检查 物品 是 否 存在 
if (!$itemId) { 
return json encode([ 
'error_code' => 1004, 


'error_message' => ' 请 选择 一 个 物品 进行 下 单 操作 ' 














]); 
} 
$itemObject = Item: :findone ($itemId); 
if (!$itemObject) { 
return json encode([ 
"error code' => 1005, 
'error_message' => ' 没 有 找到 该 物品 ， 请 选择 一 个 正确 的 物品 ' 
是 
} 


// 未 登录 用 户 需 要 先 登 录 
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} 


if (!$clientId) { 
$param = '&itemId=' . $itemId . '&needTime=' . $needTime; 
$str = baseé64 encode ($param); 
return $this->redirect([ 
"item/account', "return' => Url::toRoute(['item/request', 
"param' => $str]) 
Bs 
} 


$param = '&itemId=' . $itemId . '&needTime=' . $needTime; 
$str = base64 encode ($param); 
return $this->redirect([ 
'item/request', 'param' => $str 
Wy 


Sparam = \Yii::$app->request->get('param', false); 
if ($param) { 


$param = base64_ decode ($Sparam) 7 
parse_str ($param); 


// 检查 物品 是 否 已 经 被 他 人 下 单 
$tradeNumber = Transaction::find()->where([ 
'item id' => $itemId, 
"status' => [ 
'requested', 
vin=Fenty 
] 
])->count (); 
if ($tradeNumber > 0) { 
$message = ' 您 手 慢 了 ， 该 物品 已 被 他 人 下 单 '; 
return $this->render('message', [ 
'message' => $message 
]); 
} 


// 创建 交易 记录 

$transaction = new Transaction(); 

$transaction->id = (int) (microtime (true) * 10000) + rand(10, 99); 

Stransaction->item id = $itemId; 

S$transaction->owner_id = Item: :findone ($itemId) ->owner id; 

$transaction->timetype_id = Item::findone ($itemId) ->timetype_id; 

$transaction->renter id = \Yii::$app->session->get ('CustomerId'); 

$transaction->status = 'requested'; 

$transaction->create at = time(); 

$transaction->time = $needTime; 

$transaction->for_rent or sale = Item: :findone ($itemId)— 
>for_rent or sale; 

$transaction->start time = 0; 

Stransaction->end time = 0; 


// 更 新 物品 状态 
$itemObj = Item: :find()->where([ 
"id' => $itemId 


1)->one(); 
$itemObj->status = 'requested'; 


if ($transaction->save()) { 
if ($itemObj->save()) { 
return $this->redirect([ 
"item/request-success'" 
])7 
} else { 
$message = ' 系 统 错误 ， 请 联系 管理 员 。 错 误 代码 : 
return S$this->render('message', [ 
'message' => $message 


$message = ' 系 统 错误 ， 请 联系 管理 员 。 错 误 代码 : 800'7 
return S$this->render('message', [ 

'message' => $message 
Ds 


public function actionRequestSuccess() 
{ 


return $this->render('request-success'); 


} 


/六 
* 代理 登录 页 
* @param string $return 
6 
public function actionAccountAgent ($return = false) 
{ 
$model = new Customer () 7 
Smessage = 0107 
if (\Yii::$app->request->isPost) { 
$model->load (\Yii::$app->request->post ()); 
// 代理 用 户 登 录 
if ($model->customValidate()) { 
if ($return) { 
return $this->redirect([ 
$return 
1)s; 





} 


return $this->redirect([ 
"admin/monitor' 
i 
} 


Smessage = " 邮箱 或 密码 不 正确 ' ; 


900'; 
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学 
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if (empty($model->email) || empty($model->password)) { 
$message = ' 邮 箱 与 密码 不 能 为 空 '; 
} 
// 演 染 页 面 


return $this->render('account-agent', [ 
'model' => $model, 
'message' => $message, 
'return' => $return 

I 





} 


/太太 
* 登录 页 
过 
public function actionRccount () 
{ 
if (\Yii::$app->session->get('CustomerLogin', false)) { 
// 已 登录 用 户 跳 转 到 用 户 后 台 
return S$this->redirect([ 
"admin/index' 
ys 
} 


// 错误 信息 
$message = "'; 
// 用 户 模型 


$model = new Customer(); 


if (\Yii::$app->request->isPost) { 

// PosT 请 求 时 ， 从 POsT 请 求 中 加 载 数 据 
$model->load (\Yii::$app->request->post ()); 
if ($model->customValidate()) { 

// 登录 成 功 ， 跳 转 到 用 户 后 台 

return $this->redirect([ 

"admin/item' 

]) 7 

} 


$message = ' 邮 箱 或 密码 不 正确 。' ; 
if (empty($model->email) || empty($model->password)) { 
$message = ' 邮 箱 与 密码 不 能 为 空 。' ; 
} 
} 


// 泻 染 登录 页 面 

return $this->render('account', [ 
model' => $model, 
'message' => $message 

1); 


{ 


* 注册 页 

区 

public function actionRegister() 
// 错误 信息 
Smessage = "''; 
// 用 户 模型 


$model = new Customer (); 


if (\Yii::$app->request->isPost) { 


hn 


// 检查 两 次 密码 是 否 匹 配 
$customer = Yii::$app->request->post ('Customer'); 
if ($customer ['password'] != $customer ['confirm-password']) 
$message = ' 两 次 密码 不 匹配 '; 
return $this->render('register', [ 
'model' => $model, 
'message' => $message 
]); 
E 


// 从 PosT 请 求 中 加 载 数 据 

$model->load (\Yii::$app->request->post ()); 
$model->id = $this->createCustomerId(); 
$model->community id = 07 

$model->time = time(); 


if ($model->save()) { 
$session = \Yii::$app->session; 
// 检查 session 是 否 开启 
if (!$session->isActive) { 
// 开启 session 


$session->open(); 


} 

// 用 户 信息 保存 到 会 话 

$session->set ('CustomerId', $model->id); 
$session->set ('CustomerEmail', $model->email); 
$session->set ('CustomerName', $model->name); 
$session->set ('CustomerNumber', $model->number); 
$session->set ('CustomerLogin', true); 


// 注册 成 功 ， 跳 转 到 用 户 后 台 
return $this->redirect([ 
'admin/index' 


]); 


// 泻 染 注册 页 面 


return $this->render('register', [ 


I 


model' => $model, 
'message' => $message 


日 


{ 


i 
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/*# 
* 登 出 
*/ 
public function actionLogout () 
{ 
$session = \Yii::$app->session; 
// 检查 session 是 否 开启 
if (!$session->isActive) { 
// 开启 session 


$session->open(); 





// 清除 用 户 信息 

S$session->remove ('CustomerId'); 
$session->remove (!CustomeIrEmail')7 
$session->Iemove ('!CustomerName')7 
$session->remove ('CustomerLogin')7 


// 跳 转 到 首页 
return $this->redirect([ 
"item/index' 
I 
1 


/六 
* 使 用 说 明 
public function actionHowitworks () 
{ 
return $this->render('howitworks'); 


} 


/六 
* 生成 用 户 ID 
* @param number $country 国家 代码 
* @param number $community 社区 代码 
* @return string 
a 
protected function createCustomerIid($country = 1, $community = 0) 
{ 
$community = sprintf('%03d', $community); 
Snumber = 1; 
$model = Customer: :find()->where('id like :id'，[ 
':id' => $country . $community . '%" 
])->orderBy('id desc')->one(); 
if ($model) { 
$number = intval (substr ($model->id, 5)) + 1; 
} 
return $country . $community . sprintf('%05d', $number); 


} 


/** 
* 清除 会 话 条 件 
人 
protected function clearRelatedSession() 


\Yii::$app->session->remove('Conditions'); 
\Yii::$app->session->remove('Pricestr'); 
$app->session->remove ('Toolsstatus'); 
$app->session->remove('SportsGoodsStatus'); 
$app->session->remove ('BooksStatus'); 
$app->session->remove ('VideoGamesStatus')7 
$app->session->remove ('MusicInstrumentsSstatus'); 
$app->session->remove ('ToysAdGamesStatus'); 
$app->session->remove('Clothstatus'); 
$app->session->remove('Householdstatus'); 
$app->session->remove ('Othersstatus'); 
:$app->session->remove ('Price'); 
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private function IecordCcheckBoxStatus () 
{ 
$toolsStatus = \Yii::$app->request->post('ToolsStatus'); 
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if ($toolsStatus == '1°') { 
\Yii::$app->session->set('ToolsStatus', 1); 
} else { 


\Yii::$app->session->remove('ToolsStatus'); 


$sportsGoodsStatus = \Yii app->request->post ('SportsGoodsStatus'); 
if ($sportsGoodsSstatus i 

\Yii::$app->session->set ('SportsGoodsStatus', 1); 
} else { 

\Yii::$app->session->remove('SportsGoodsStatus'); 








| 
SbooksStatus = \Yii::$app->request->post('Booksstatus'); 
if ($booksStatus == '1°') { 
\Yii::$app->session->set ('BooksStatus', 1); 
} else { 


\Yii::$app->session->remove('BooksStatus'); 


$videoGamesStatus = \Yii::$app->request->post('VideoGamesStatus'); 





if ($videoGamesStatus == '1°') { 
\Yii::$app->session->set ('VideoGamesStatus', 1); 
} else { 
\Yii::$app->session->remove('VideoGamesStatus'); 
lh 
SmusicInstrumentsStatus = \Yii::$app->request->post ('MusicInstrumentsSstatus'); 
if ($musicInstrumentsstatus == "1') { 
\Yii::$app->session->set ('MusicInstrumentsSstatus', 1); 
} else { 


\Yii::$app->session->remove('MusicInstrumentsstatus'); 
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$toysAdGamesStatus = \Yii::$app->request->post('ToysAdGamesStatus'); 
if ($toysAdGamesstatus == '1') { 
\Yii::$app->session->set('ToysAdGamesStatus', 1); 
} else { 
\Yii::$app->session->remove('ToysAdGamesStatus'); 








上 

$clothstatus = \Yii::$app->request->post('Clothstatus'); 

if ($clothStatus == '1') { 
\Yii::$app->session->set('Clothstatus', 1); 

} else { 


\Yii::$app->session->remove('Clothstatus'); 


} 





N ShouseholdSstatus = \Yii::$app->request->post('Householdstatus'); 
if ($householdstatus ER 
\Yii::$app->session->set('Householdstatus', 1); 
} else { 
\Yii::$app->session->remove('Householdstatus'); 





} 





$othersstatus = \Yii::$app->request->post('Othersstatus'); 


IE ($othersstatus == "1') { 
\Yii::$app->session->set ('OthersStatus', 1); 
} else { 


\Yii::$app->session->remove('OthersSstatus'); 


<!DOCTYPE html> 
<html lang="zh-CN"> 
<head> 
<meta charset="utf-8"> 
<meta http-equiv="X-UA-Compatible" content="IE=edge"> 
<meta name="viewport" content="width=device-width, initial-scale=1"> 


<title> 社 区 市 场 系统 </title> 


<link href="css/bootstrap.css" rel="stylesheet"> 


<I==[if 1t TE 91> 
<script 
src="https://cdn.bootcss.com/htmlS5shiv/3.7.3/html5shiv.min.js"></script> 
<script 
src="https://cdn.bootcss.com/respond.js/1.4.2/respond.min.js"></script> 
<![endif]--> 
</head> 
<body> 
<div class="container"> 
<div class="row"> 
<div class="col-xs-12"> 
<hl class="text-center"> 社 区 市 场 系统 </h1> 
<hr/> 


</div> 
</div> 
<div class="row"> 
<div class="col-xs-12"> 
<div style="max-width: 320px; margin: 20px auto;"> 
<div class="form-group"> 
<label for="username"> 用 户 名 </label> 
<input type="text" name="username" value="<?= $loginUser->name ?3? 
'! 2>" class="form-control"” id="username" placeholder=" 用 户 名 "> 
</div> 
<div class="form-group"> 
<label for="password"> 密 码 </label> 
<input type="password" name="password" value="" class="form-— 
control" id="password" placeholder=" 密 码 "> 
</div> 
<br> 
<div class="form-group text-center"> 
<button class="btn btn-primary btn-lg" id="login" type="button" 
style="min-width: 300px;"> 登 录 </button> 
</div> 
</div> 
</div> 
</div> 
<div class="row"> 
<div class="col-xs-12 text-center"> 
<hr/> 
<h3> 用 户 列表 (默认 密码 : 123456) </h3> 
<br> 
<div class="text-muted"> 
<?php foreach ($users as $user) { ?> 
<p><?= $user['name'] ?> (<?= $user['role'] ?>)</p> 
<?php } ?> 
</div> 
</div> 
</div> 
</div> 


<script src="js/jquery-3.2.1.js"></script> 
<script type="text/javascript"> 
$(function() { 
$('#login') .on('click', function(e) { 
Var username = $('#username') .val (); 


if ($.trim(username) == '') { 
$('#username') .focus ()7 
alert (' 请 输入 用 户 名 ' ) ; 
return; 
} 
var password = $('#password') .val (); 
if ($.trim(password) == "'') { 
$('#password') .focus () 7 
alert (' 请 输入 密码 ') ; 
return; 


1 
$.post('index.php', { 
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和 CEORSUEGOOYDO 
username: USernamer 
password: password 
}, function(result) { 
if (result && result.error) { 
alert (result .error); 
return; 
} 
location.href = 'index.php'; 
}, 'json'); 





Pha 
</script> 
</body> 
</html> 


23.4.4 用 户 后 台 控 制 器 


AdminController php 为 用 户 后 台 控制 器 ， 主 要 帮助 用 户 发 布 与 管理 物品 和 需求 ， 以 及 查 
看 订单 。 对 于 代理 用 户 ， 主 要 用 于 更 新 交易 状态 。 代 码 如 下 : 


<?php 
namespace app\controllers; 


use app\models\Customer; 

use app\models\Item; 

use app\models\ItemImg; 

use app\models\Need; 

use app\models\NeedImg; 

use app\models\Transaction; 

use app\models\UploadForm; 

Use Yiis 

use yii\data\ActiveDataProvider; 
use yii\data\Pagination; 

use yii\web\NotFoundHttpException; 
use yii\web\UploadedFile; 


/*# 
* 用 户 后 台 控制 器 
wd 
class AdminController extends \yii\web\Controller { 
/** 布局 */ 
public $layout = 'admin.php'; 


/** 
* 用 户 后 台 首 页 
区 
public function actionIndex() { 
// 跳 转 到 我 的 物品 页 
return Sthis->redirect([ 
"admin/item' 
这 


第 

1 3 

/** 司 

* 发 布 物品 页 项 

上 目 

public function actionItemAdd() { 六 
if (!\Yii::$app->session->get('CustomerLogin', false)) { 

// 未 登录 用 户 跳 转 到 登录 页 | 

return S$this->redirect([ 

"item/account' 区 

I 年 

区 

市 

// 物品 模型 要 

统 


$model = new Item(); 


if (\Yii::$app->request->isPost) { 
// 从 PosT 请 求 加 载 数据 
$model->load (\Yii::$app->request->post ()); 
$model->id = $this->createItemId(); 
$model->owner_id = \Yii::$app->session->get('CustomerId'); 
$model->community_ id = 07 
$model->rent_ from = time(); 





// 格式 化 数据 
if ($model->rent to != "") { 

$model->rent to = strtotime ($model->rent to); 
} else { 





$model->rent to = 0; 

} 

if ($model->deposit == "") { 
$model->deposit = 0; 


if ($model->save()) { 
if (Yii::$app->session->has('pic')) { 

// 保存 物品 图 片 

$imagesArr = Yii::$app->session->get ('pic'); 

foreach ($imagesArr as $image) { 
$itemImg = new ItemImg (); 
$itemImg->item id = $model->id; 
$itemImg->images = $image; 
$itemImg->save () 7 

| 


Yii::$app->session->remove('pic'); 


} 
// 跳 转 到 我 的 物品 页 


return $this->redirect(['admin/item’']); 


. 
// 格式 化 数据 


if ($model->rent from == 0) { 
$model-—>rent from = ""; 


} 
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if ($model->rent to Oe 
$model-—>rent to = ""; 
有 
if ($model->rent from) { 
$model->rent from = date('m/d/y', $model->rent from); 
} 
if ($model->rent to) { 
$model->rent to = date('m/d/y', $model->rent to); 
} 





} else { 
Yii::$app->session->remove('pic'); 
// 演 染 发 布 物品 页 


return $this->render('item-add', [ 
'model' => $model 


public function actionNeedAadd() { 
if (!\Yii::$app->session->get('CustomerLogin', false)) { 
// 未 登录 用 户 跳 转 到 登录 页 
return $this->redirect([ 
‘item/account', 'note' => 1 
my 
} 


// 需求 模型 


$model = new Need(); 


if (\Yii::$app->request->isPost) { 
// 从 PosT 请 求 加 载 数据 
$model->load (\Yii::$app->request->post ()); 
$model->id = $this->createNeedId(); 
$model->owner_id = \Yii::$app->session->get('CustomerId'); 
Smode1->community id = 07 
$model->need from = strtotime ($model->need from); 


if ($model->save()) { 
if (Yii::$app->session->has('pic')) { 

// 保存 需求 图 片 

$imagesArr = Yii::$app->session->get ('pic'); 

foreach ($imagesArr as $image) { 
SneedImg = new NeedImg (); 
SneedImg->need id = $model->id; 
SneedImg->images = $image; 
SneedImg->save () 7 





} 
Yii::$app->session->remove('pic'); 
} 


return $this->redirect(['admin/need']); 


第 
1 S| 
if ($model->need from) { 章 | 
$model->need from = date('m/d/y', $model->need from); 
时 项 
} else { 是 
Yii::$app->session->remove('pic'); 六 
j 
// 泻 染 发 布 需求 页 | 
return $this->render('need-add', [ 
'model' => $model 社 
JJ 区 
市 
场 
系 
* 我 的 物品 页 
SE/ 





public function actionItem() { 
if (!\Yii::$app->session->get('CustomerLogin', false)) { 
// 未 登录 用 户 跳 转 到 登录 页 
return S$this->redirect([ 
'item/account' 


} 
// 查找 我 发 布 的 物品 


$query = Item: :find()->where([ 
'owner_id' => \Yii::$app->session->get('CustomerId') 
]) 7 
$dataProvider = new ActiveDataProvider([ 
"query' => $query 
]) 7 


// 使 用 翻 页 组 件 
$pages = new Pagination([ 
"totalCount' => $query->count(),'pageSize' => '5' 
]) 7 
$data = $query->offset (Spages->offset)->1limit(S$pages->1limit)->al1()7 





// 泻 染 页 面 
return $this->render('item', [ 

'dataProvider' => $dataProvider, 'data' => $data, 'pages' => $pages 
Wn 


大 大 


* 我 的 需求 页 

jh 

public function actionNeed() { 

if (!\Yii::$app->session->get('CustomerLogin', false)) { 
// 未 登录 用 户 跳 转 到 登录 页 
return $this->redirect([ 
"item/account" 

a 
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| 


// 查找 我 发 布 的 需求 
$query = Need: :find()->where([ 
"owner_id' => \Yii::$app->session->get('CustomerId') 
I 
$dataProvider = new ActiveDataProvider([ 
'query' => $query 
1); 


// 使 用 翻 页 组 件 
$pages = new Pagination([ 
"totalCount' => $query->count(),'pageSize' => '5' 





1); 
$data = $query->offset ($pages->offset)->1imit($Spages->1imit)->al1() 7 


// 泻 染 页 面 
return $this->render('need', [{ 
'dataProvider' => $dataProvider,'data' => $data,'pages' => $pages 
])7 
二 


/六 

* 查看 物品 页 

* @param string $id 

六 

public function actionItemView($id) { 
return $this->render('item-view', [ 

'model' => $this->findItemModel ($id) 

1); 

1 


/a** 

* 查看 需求 页 

* @param string $id 

*W 

public function actionNeedView($id) { 
return $this->render('need-view', [ 

model' => $this->findNeedModel ($id) 

]) 7 

1 


/*# 

* 删除 物品 

六 

* @param string $id 

wf 

public function actionItemDelete($id) { 
// 获取 物品 
$model = $this->findItemModel ($id); 
if ($model->status != ‘'available') { 


// 如 果 物 品 已 下 单 ， 不 能 再 删除 ， 跳 转 到 查看 页 
return Sthis->redirect([ 
"admin/item-view', "id => $id 


第 

]) 7 B 

} 章 
// 删除 物品 项 
$model->delete (); 且 
济 


// 获取 物品 图 片 
$itemImg = ItemImg: :find()->where([ 
"item id' => $id 
])->all(); 
if ($itemImg) { 
// 删除 图 片 文件 
foreach ($itemImg as $img) { 
$url = substr ($img->images, 1); 
@unlink ($url); 


芝 洲 总 填 网 序 掉 首 一 一 


} 
} 
// 删除 图 片 数据 


ItemImg: :deleteAll([ 
"item id' => $id 


WO 
// 跳 转 到 我 的 物品 页 


return $this->redirect([ 
"radmin/item' 
I 
} 





/六 

* 删除 需求 

* @param string $id 

public function actionNeedDelete($id) { 
// 删除 需求 
$this->findNeedModel ($id) ->delete(); 
// 获取 需求 图 片 
$needImg = NeedImg::find()->where([ 

"need id => $id 

1])->all(); 


if ($needImg) { 
// 删除 需求 图 片 文件 
foreach ($needImg as $img) { 
$url = substr ($img->images, 1); 
@unlink ($url); 
} 


和 

// 删除 图 片 数 据 

NeedImg: :deleteRAll([ 
need _ id' => $id 

I 


// 跳 转 到 我 的 需求 页 


return $this->redirect([ 
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"admin/Vneed' 
Ws 
} 
/ 
* 更 新 物品 页 
* @param string $id 
和 
public function actionItemUpdate (Sid) { 
// 获取 物品 
$model = $this->findItemModel ($id); 
if ($model->status != 'available') { 
// 如 果 物 品 已 下 单 ， 不 能 再 编辑 ， 跳 转 到 查看 页 
return S$this->redirect([ 
"admin/item-view', 'id' => $id 
1); 
} 
// 获取 物品 图 片 


$imageArr = ItemImg::find()->where([ 
"item id' => $id 
])->asArray() ->all(); 


if (Yii::$app->request->isPost) { 
// 从 PosT 请 求 加 载 数 据 
if ($model->load (Yii::$app->request->post())) { 
$model->rent_ from = time(); 


// 格式 化 数据 
if ($model->rent to != "") { 

$model->rent to = strtotime ($model->rent to); 
} else { 


$model->rent to = 0; 

} 

if ($model->deposit == "") { 
$model->deposit = 0; 

} 


if ($model->save()) { 
// 跳 转 到 物品 查看 页 
return $this->redirect([ 
"admin/item-view', 'id' => $model->id 


1); 


} 


// 格式 化 数据 

if ($model->rent from == 0) { 
Smode1->rent_from = ""; 

} 

if ($model->rent to == 0) { 
$model->rent to = ""; 

| 


if ($model->rent from) { 
$model->rent_ from = date('m/d/y', $model->rent from); 


if ($model->rent to) { 
$model->rent to = date('m/d/y', $model->rent to); 
上 


Yii::$app->session->remove('pic'); 


// 泻 染 物品 更 新 页 
return $this->render('item-update', [ 
'model' => $model,'imageArr' => $imageArr,'id' => $id 
I 
} 


/太太 
* 更 新 需求 页 
* @param string $id 
public function actionNeedUpdate ($id) 
{ 

// 获取 需求 


$model = $this->findNeedModel ($id); 

// 获取 需求 图 片 

$imageArr = NeedImg::find()->where([ 
'need id' => $id 

])->asArray()->all(); 


if (Yii::$app->request->isPost) { 
// 从 PosT 请 求 加 载 数据 
if ($model->load (Yii::$app->request->post())) { 
$model->owner_id = \Yii:: 
$model->community_id = 07 
$model->need from = strtotime ($model->need from); 





if ($model->save()) { 
if (Yii::$app->session->has('pic')) { 

// 保存 需求 图 片 

$imagesArr = Yii::$app->session->get ('pic'); 

foreach ($imagesArr as $image) { 
$needImg = new NeedImg (); 
$needImg->need id = $model->id; 
$needImg->images = $image; 
$needImg->save (); 





} 


Yii::$app->session->remove('pic'); 


} 

// 跳 转 到 查看 页 

return $this->redirect([ 
"admin/need-view', 'id' => $id 

a 


| 


Yii::$app->session->remove('pic'); 


// 泻 染 页 面 


$app->session->get ('CustomerId'); 


A 


将 三 尖山 8z 小 


学 
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return $this->render('need-update', [ 
'model' => $model, 
"imageArr' => $imageArr, 


Nd 
I 
} 
/*# 
* 我 的 交易 记录 页 
x 


public function actionTransaction() { 
if (!\Yii::$app->session->get('CustomerLogin', false)) { 
// 未 登录 用 户 跳 转 到 登录 页 
return S$this->redirect([ 
'item/account' 
LY 





} 
// 查找 我 的 交易 记录 


$query = Transaction::find()->where([ 
'renter_id' => \Yii::$app->session->get('CustomerId') 





1); 

$dataProvider = new RARctiveDataProvider([ 
"query' => $query 

I 


// 使 用 翻 页 组 件 
$pages = new Pagination([ 
"totalCount' => $query->count(), 
'pageSize' => '5°' 
1); 
$data = $query->offset ($pages->offset)->limit($pages->limit)->all (); 


// 演 染 页 面 

return S$Sthis->render ('transaction'，[ 
"dataProvider' => $dataProvider, 
'data' => $data, 
'pages' => $pages 


闪闪 
* 交易 一 览 页 
Ea 
public function actionMonitor () 
{ 
if (!\Yii::$app->session->get('CustomerLogin', false)) { 
// 未 登录 用 户 跳 转 到 登录 页 
return $this->redirect([ 
'item/account-agent' 
1); 
} 


// 查找 所 有 交易 记录 


} 


$query = Transaction::find(); 

$dataProvider = new ActiveDataProvider([ 
'query' => $query 

1 


// 使 用 翻 页 组 件 
$pages = new Pagination([ 
"totalCount' => $query->count (), 
"pageSize' => "5" 
Ds 
$data = $query->offset ($pages->offset)->limit ($pages->limit)->all(); 


// 泻 染 页面 

return $this->render('monitor', [ 
"dataProvider' => $dataProvider, 
'data' => $data, 
'pages' => $pages 

1); 


We 


* 交易 更 新 页 
* @param string $id 


en 


public function actionMonitorUpdate ($id) 


{ 


if (!\Yii::$app->session->get('CustomerLogin', false)) { 
// 未 登录 用 户 跳 转 到 登录 页 
return $this->redirect([ 
1" item/account-agent'" 
下 有 
} 


// 获取 交易 模型 


$model = $this->findMonitorModel ($id); 


if (Yii::$app->request->isPost) { 
if ($model->load (Yii::$app->request->post())) { 


if ($model->status == 'in-rent' && !$model->start time) { 
// 租赁 开始 时 间 
Smode1->start_time = time(); 

} 

if ($model->status == 'complete' && !$model->end time) { 
// 交易 结束 时 间 
$model->end time = time(); 

} 

if ($model->save()) { 


if ($model->status != 'requested') { 
$item = Item: :findone ($model->item id); 
if ($item) { 
if ($model->for rent or sale rent') { 
$item->status = $model->status == 'complete' ? 
"available' : 'requested'; 
$item->save () 7 





im 加 由 tz 典 司 


二 
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} else if ($model->for rent or sale == 'sale' && 
$model->status == 'complete') { 
$item->status = $model->status; 
$item->save (); 





} 
!: 
Sthis->redirect([ 
"admin/monitor' 
]) 7 
} 
和 
} 
// 泻 染 页 面 


return $this->render('monitor-update', [ 
'model' => $model 


public function actionUploadImage() { 
$uploadFormModel = new UploadForm(); 
$id = Yii::$app->request->get('id', false); 


if (Yii::$app->request->isAjax) { 
$uploadFormModel->file = UploadedFile::getInstance($uploadFormModel, 
'pic'); 

if ($uploadFormModel->file && $uploadFormModel->validate()) { 
// 保存 图 片 
$file = time() . '.' . $uploadFormModel->file->extension; 
$uploadFormModel->file->saveAs('upload/picture/' . $file); 
$url = '/upload/picture/' . $file; 


// 设置 图 片 到 会 话 

if (Yii::$app->session->has('pic')) { 
Stmp = Yii::$app->session->get ('pic'); 
array_push($tmp, $url); 
Yii::$app->session->set('pic', $tmp); 

} else { 
$tmp = array ($url); 
Yii::$app->session->set ('pic', $tmp); 

} 

if ($id) { 
// 生成 图 片 
$itemImg = new ItemImg () 7 
$itemImg->item id = $id; 
SitemImg->images = $url; 
SitemImg->save (); 
SneedImg = new NeedImg (); 
SneedImg->need id = $id; 
SneedImg->images = $url; 
SneedImg->save (); 


第 
} S 
// 返回 图 片 地 址 | 
return json encode (array( 项 
"imgUrl' => $url 目 
二 头 
} 训 
|! 
} 
并 
太太 发 
* 删除 图 片 划 
Wk 
public function actionDeleteImg() { 下 
// 获取 图 片 系 
$img = \Yii::$app->request->post ('img'); 统 


if ($img == null) { 
$img = \Yii::$app->request->get ('img'); 
} 
$file = \Yii::$app->basePath . "/web" . $img; 


if (\Yii::$app->session->has('pic')) { 
$tmp_new = []; 
$tmp = \Yii::$app->session->get ('pic'); 
if (!empty($tmp)) { 
foreach ($tmp as S$record) { 
if (strpos($record, $img) === false) { 
array_push($tmp_new, S$record); 





1 
} 
1 
NYii::S$Sapp->session->set('pic'，S$Stmp_new) 7 


} 
// 删除 物品 图 片 


$itemImg = ItemImg::find()->where([ 
'images' => $img 

]) ->one () 7 

if ($itemImg) { 
$itemImg->delete(); 

} 


// 删除 需求 
SneedImg = NeedImg::find()->where([ 
'images' => $img 
])->one () 7 
if ($needImg) { 
SneedImg->delete () 7 





} 


@unlink ($file); 
return $file; 


大 大 


* 我 的 档案 页 
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public function actionProfile() { 
if (!\Yii::$app->session->get('CustomerLogin', false)) { 
// 未 登录 用 户 跳 转 到 登录 页 
return $this->redirect([ 
"item/account' 
Wm 
} 


// 错误 信息 
Smessage = "''; 


// 用 户 信息 


$model = Customer::findone (Yii::$app->session->get ('CustomerId')); 


$errors = array(); 
if (Yii::$app->request->isPost) { 
$post = Yii::$app->request->post () 7 
if (empty($post ['Customer'] ['change-password'])) { 
unset($post ['Customer'] ['password']); 


} 
// 从 PosT 请 求 加 载 数 据 
if ($model->load($post)) { 
if (!empty($post ['Customer'] ['change-password'])) { 
// 验证 原 密 码 
if (!$model->customValidate()) { 
$model->addError ('password'，' 请 输入 正确 的 原 密码 ' ) ; 
} else { 
if ($post ['Customer'] ['new-password'] == '') { 


$errors ['new-password'] = ' 新 密码 不 能 为 空 '; 





} 
if ($post ['Customer'] ['new-password'] != $post 
['Customer'] ['confirm-password']) { 
$errors ['confirm-password'] = "两 次 密码 不 匹配 ' ; 
} 
if (empty($errors)) { 
$model->password = $post ['Customer'] ['new-password']; 
} 
} 
} 
if (empty($errors) && $model->save()) { 
$message = ' 更 新 成 功 '; 
} 


} 
// 演 染 页 面 


return $this->render('profile', [ 
'model' => $model, 'message' => $message, 'errors' => $errors 
]) 7 
} 


/*# 

* 创建 物品 ID 

* @param number $country 国家 代码 

* @param number $community 社区 代码 

wt 

protected function createItemId($country = 1, $community = 0) { 
$community = sprintf('%03d', $community); 


全 
国 国 
a 


第 
$number = 1; S| 
$model = Item: :find()->where('id like :id', [ 二 | 

':id' => $country . $community . '%" 司 
])->orderBy('id desc')->one(); 项 
if ($model) { 目 
$number = intval (substr ($model->id, 5)) + 1; 妈 

3 
return $country . $community . sprintf('%05d', $number); | 

} 

天 
太太 发 
* 创建 需求 ID 社 
* @param number $country 国家 代码 
* @param number $community 社区 代码 场 
#/ 系 
protected function createNeedId($country = 1, $community = 0) { 统 


$community = sprintf('%03d', $community); 
$number = 1; 
$model = Need: :find()->where('id like :id', [ 
':id' => $country . $community . '%" 
])->orderBy('id desc')->one(); 
if ($model) { 
$number = intval(substr ($model->id, 5)) + 1; 





AG 


l 


return $country . $community . sprintf('%05d', $number); 





太太 


* 获取 物品 模型 


* @param integer $id 
二 
protected function findItemModel ($id) { 
if (($model = Item::findone($id)) 
return $model; 
} else { 
throw new NotFoundHttpException(' 请 求 的 物品 不 存在 '); 


null) { 





} 
} 


/* 这 
* 获取 需求 模型 


* @param integer $id 
本 
protected function findNeedModel ($id) 1{ 
if (($model = Need::findone($id)) ! 
return $model; 
} else { 
throw new NotFoundHttpException(' 请 求 的 需求 不 存在 '); 





null) { 


1 


妈妈 


* 获取 交易 模型 


* @param integer $id 
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nh 
protected function findMonitorModel ($id) 
{ 
if (($model = Transaction::findone ($id)) !== null) { 
return $model; 
} else { 
throw new NotFoundHttpException(' 请 求 的 交易 不 存在 '); 


23.4.5 ”数据 模型 的 文件 


Models 文件 夹 下 包含 了 很 多 数据 模型 文件 ， 主 要 实现 数据 表 category( 物 品 分 类 )、 
comment( 留 言 )、customer( 用 户 )、item( 物 品 )、item_img( 物 品 图 片 )、need( 需 求 )、need_img( 需 
求 图 片 )、timetype( 时 间 类 型 )、transaction( 交 易 ) 和 type( 交 易 方 式 ) 的 模型 。 

下 面 查看 一 下 Customer.php( 用 户 数据 模型 ) 的 代码 : 


<?php 





namespace app\models; 


use Yii; 
/六 娘 
* 用 户 数据 模型 
和 
* @property integer $id 
* @property string $name 
* Q@property string $email 
* @property string $password 
* @property string $extkey 
* @property integer $community_id 
* @property integer $customer type 
Ch 
class Customer extends \yii\db\ActiveRecord 
/** 
* @inheritdoc 
sd 


public static function tableName() 
{ 
return 'customer'; 


} 
/** 


* @inheritdoc 
< 
public function rules() 
{ 
return [ 
[['id', 'name', "email', 'password', 'community id','time'], 'required'], 
[['id', 'community_ id','customer type', 'time'] ,'integer'], 


[['name'，'email'，'password'v'number']，'string'， 'max' 
//[['name'], 'unique'], 

[['email'], ‘'unique'], 

[['email'], ‘email'], 

//[['number'], "unique'], 


} 
六 


* @inheritdoc 

of 

public function attributeLabels () 

{ 

return [ 

"mid" =>, "LIDYy 
'name' => ' 名 字 '， 
remail' =>' 邮 箱 '， 
"password' => ' 密 码 '， 
"extkey' => 'Ext Key', 
"community_id' => ' 社 区 '， 
'time' => ' 时 间 '， 
'number' => ' 手 机 号 码 '， 


} 
xy 


* @inheritdoc 

a 

public function beforeSave ($insert) 

{ 
$this->password = md5 ($this->password); 
return true; 


/六 
* 用 户 登 录 验 证 ， 检 查 用 户 名 与 密码 是 否 匹 配 
* @return boolean 
二 

public function customValidate () 

$customer = static::findone([ 
'email' => $this->email 
1); 


if ($customer) { 
if ($customer->password === md5($this->password)) { 
$session = Yii::$app->session; 


// 检查 session 是 否 开启 
if (!$session->isActive) { 

// 开启 session 

$session->open(); 
3 
$session->set ('CustomerId', $customer->id); 
$session->set ('CustomerEmail', $customer->email); 
$session->set ('CustomerName', $customer->name); 
$session->set ('CustomerLogin', true); 


0 


a 
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$session->set ('isAgent', $customer->customer type == 99); 
return true; 
FE 
} 


return false; 
} 
} 
由 于 社区 市 场 系统 的 文件 太 多 ， 这 里 只 讲解 了 核心 的 文件 。 对 于 其 他 文件 ， 读 者 可 以 参 
照 本 书 的 源 文件 进行 查 。 








\ 23.5 系统 测试 


下 面 测 试 社区 市 场 系统 的 功能 。 有 具体 操作 步骤 如 下 。 
在 浏览 器 中 输入 社区 交易 系统 的 主页 的 地 址 : http://localhost/code/ch23/web/ 
index.php， 结 果 如 图 23-5 所 示 。 
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23-5 社区 市 场 主页 面 


EZ 单 击 “ 使 用 说 明 ” 按 钮 ， 即 可 查看 社区 物品 分 享 的 流程 ， 如 图 23-6 所 示 。 
TB 在 主页 面 中 单 击 “ 代 理 ” 链 接 ， 即 可 进入 代理 登录 页 面 ， 输 入 代理 的 名 称 和 密 
码 ， 如 图 23-7 所 示 。 


二 
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社区 分 享 ， 只 需 四 步 ! 
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图 23-6 ”查看 社区 物品 分 享 的 流程 
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图 23-7 代理 登录 页 面 


EE 单 击 “登录 ”按钮 ， 即 可 进入 代理 的 后 台 页 面 ， 可 以 查看 系统 的 交易 记录 和 我 


的 档案 信息 ， 如 图 23-8 所 示 。 
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23-8 ”代理 后 台 页 面 
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EECUD9 单 击 “ 登 出 ”链接 ， 即 可 退回 到 系统 主页 面 中 。 单 击 “注册 ”链接 ， 即 可 进入 
社区 用 户 注册 页 面 ， 如 图 23-9 所 示 。 输 入 注册 信息 后 ， 单 击 “ 注 册 ” 按 钮 ， 即 可 完 



























































成 用 户 注册 的 操作 。 
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图 23-9 注册 页 面 


EERUD 注册 成 功 后 ， 即 可 在 主页 面 中 单 击 “ 登 录 ” 链 接 ， 进 入 用 户 登 录 页 面 ， 输 入 用 
户 名 和 密码 后 ， 单 击 “ 登 录 ” 按 钮 即 可 ， 如 图 23-10 所 示 。 




















= 口 x 
© IEC * 国 Oe 
| 文 #p。 映 E 坪 看 V| 加 天 (A| 工具 m) 帮助 (H| 

的 社区 市 场 您 要 什么 ? 人 
| ao 去 ee 
登 对 账户 
| CO 
| v00eol 一 
成 100% 








23-10 社区 登录 页 面 


EC 登录 成 功 后 ， 进 入 用 户 的 后 台 管理 页 面 ， 默 认 选 择 “ 我 的 物品 ”选项 ， 可 以 看 
出 ， 用 户 还 没有 发 布 物品 ， 如 图 23-11 所 示 。 
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图 23-11 社区 用 户 管理 页 面 有 


EEIB 进入 “发 布 您 的 物品 ”页 面 ， 根 据 页 面 的 提示 ， 输 入 物品 的 相关 信息 即 可 ， 如 
图 23-12 所 示 。 
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图 23-12 “发 布 您 的 物品 ”页 面 


攻 于 BY 如 果 需 要 上 传 图 片 信息 ， 可 以 单 击 “上 传 ”按钮 ， 在 打开 的 “选择 要 加 载 的 文 
件 ” 对 话 框 中 选择 需要 上 传 的 图 片 ， 如 图 23-13 所 示 。 
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图 23-13 “选择 要 加 载 的 文件 ”对 话 框 


EEC 单 击 “打开 ”按钮 ， 返 回 到 “发 布 您 的 物品 ”页 面 ， 单 击 “ 发 布 物品 ”按钮 ， 
弹出 确认 发 布 信息 对 话 框 ， 如 图 23-14 所 示 。 
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图 23-14 ”确认 发 布 信息 


EEC 单 击 “ 确 认 ” 按 钮 ， 即 可 完成 物品 的 发 布 操作 。 在 物品 列表 中 可 以 看 到 刚刚 发 
布 的 物品 信息 ， 用 户 可 以 编辑 和 删除 发 布 的 物品 信息 ， 如 图 23-15 所 示 。 
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23-15 成功 发 布 物品 信息 


ECEDP) 在 左 侧 列表 中 选择 “我 的 需求 ”选项 ， 进 入 “我 的 需求 ”页 面 ， 如 图 23-16 所 示 。 
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23-16 “我 的 需求 ”页 面 


EEE 单 击 “ 添 加 ”按钮 ， 进 入 “发 布 您 的 需求 ”页 面 ， 根 据 页 面 提示 输入 需求 信 


息 ， 如 图 23-17 所 示 。 
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图 23-17 “发 布 您 的 需求 ”页 面 


EEC 单 击 “发 布 需 求 ”按钮 ， 返 回 “ 我 的 需求 ”页 面 ， 即 可 完成 发 布 物品 需求 的 操 
作 。 在 需求 列表 中 可 以 看 到 刚刚 发 布 的 需求 信息 ， 用 户 可 以 编辑 和 删除 发 布 的 需求 


信息 ， 如 图 23-18 所 示 。 
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图 23-18 “我 的 需求 ”页 面 


EEC 在 左 侧 列表 中 选择 “我 的 交易 ”选项 ， 进 入 “交易 记录 ”页 面 ， 如 图 23-19 所 示 。 
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图 23-19 “交易 记录 ”页 面 


ECEDy 在 左 侧 列表 中 选择 “我 的 档案 ”选项 ， 进 入 “编辑 档案 ”页 面 ， 
名 称 、 邮 箱 、 手 机 号 码 和 密码 等 信息 ， 如 图 23-20 所 示 。 
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图 23-20 “编辑 档案 ”页 面 
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息 ， 如 图 23-21 所 示 。 章 
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图 23-21 社区 市 场 主页 面 
EEaB 如 果 其 他 用 户 单 击发 布 的 物品 转让 信息 ， 即 可 进入 物品 转让 详情 页 面 ， 如 图 23-22 
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23-22 ”物品 转让 详情 页 面 
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EC 单 击 “我 的 问题 ”按钮 ， 即 可 在 弹出 的 页 面 中 输入 需要 咨询 的 问题 ， 单 击 “ 提 
问 ”按钮 ， 即 可 提出 疑问 ， 如 图 23-23 所 示 。 
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可 以 降价 吗 ? 
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图 23-23 “我 的 问题 ”页 面 


EEED 如 果 没 有 疑问 ， 可 以 在 物品 转让 详情 页 面 中 单 击 “ 购 买 ”按钮 ， 打 开 确 认购 买 
信息 页 面 ， 如 图 23-24 所 示 。 
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图 23-24 ”确认 购买 信息 页 面 


EEZ 单 击 “ 确 认 ” 按 钮 ， 即 可 成 功 下 单 。 单 击 “ 确 定 ” 按 钮 ， 即 可 完成 购买 物品 操 
作 ， 如 图 23-25 所 示 。 
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您 已 成 功 下 单 ! 社区 代理 将 会 联系 物 主 
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23-25 成功 下 单 页 面 


EPZ 此 时 以 代理 账号 登录 ， 即 可 在 交易 记录 中 查看 到 社区 用 户 下 的 订单 ， 如 图 23-26 
所 示 。 


使 400 







人 给 社区 市 场 你 需要 什么 9 主人 t 和 下 
Eo) 旧 
我 的 人 过 
训 
交易 记录 1 
mt 区 
物品 名 先王 下 单方 电话 A 了 ES 间 社 
九 成 新 电脑 转让 无 涯 圭 角 天 天 学 习 13012365476 ”出 告 已 下 章 -~ 四 了 区 
图 23-26 ”查看 交易 记录 对 





401 合 


